clang 23.0.0git
CIRGenRecordLayoutBuilder.cpp
Go to the documentation of this file.
1//===----------------------------------------------------------------------===//
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 compute the layout of a record.
10//
11//===----------------------------------------------------------------------===//
12
13#include "CIRGenBuilder.h"
14#include "CIRGenModule.h"
15#include "CIRGenTypes.h"
16
18#include "clang/AST/Decl.h"
19#include "clang/AST/DeclCXX.h"
24#include "llvm/Support/Casting.h"
25
26#include <memory>
27
28using namespace llvm;
29using namespace clang;
30using namespace clang::CIRGen;
31
32namespace {
33/// The CIRRecordLowering is responsible for lowering an ASTRecordLayout to an
34/// mlir::Type. Some of the lowering is straightforward, some is not.
35// TODO: Detail some of the complexities and weirdnesses?
36// (See CGRecordLayoutBuilder.cpp)
37struct CIRRecordLowering final {
38
39 // MemberInfo is a helper structure that contains information about a record
40 // member. In addition to the standard member types, there exists a sentinel
41 // member type that ensures correct rounding.
42 struct MemberInfo final {
43 CharUnits offset;
44 enum class InfoKind { VFPtr, Field, Base, VBase } kind;
45 mlir::Type data;
46 union {
47 const FieldDecl *fieldDecl;
48 const CXXRecordDecl *cxxRecordDecl;
49 };
50 MemberInfo(CharUnits offset, InfoKind kind, mlir::Type data,
51 const FieldDecl *fieldDecl = nullptr)
52 : offset{offset}, kind{kind}, data{data}, fieldDecl{fieldDecl} {}
53 MemberInfo(CharUnits offset, InfoKind kind, mlir::Type data,
54 const CXXRecordDecl *rd)
55 : offset{offset}, kind{kind}, data{data}, cxxRecordDecl{rd} {}
56 // MemberInfos are sorted so we define a < operator.
57 bool operator<(const MemberInfo &other) const {
58 return offset < other.offset;
59 }
60 };
61 // The constructor.
62 CIRRecordLowering(CIRGenTypes &cirGenTypes, const RecordDecl *recordDecl,
63 bool packed);
64
65 /// Constructs a MemberInfo instance from an offset and mlir::Type.
66 MemberInfo makeStorageInfo(CharUnits offset, mlir::Type data) {
67 return MemberInfo(offset, MemberInfo::InfoKind::Field, data);
68 }
69
70 // Layout routines.
71 void setBitFieldInfo(const FieldDecl *fd, CharUnits startOffset,
72 mlir::Type storageType);
73
74 void lower(bool NonVirtualBaseType);
75 void lowerUnion();
76
77 /// Determines if we need a packed llvm struct.
78 void determinePacked(bool nvBaseType);
79 /// Inserts padding everywhere it's needed.
80 void insertPadding();
81
82 void computeVolatileBitfields();
83 void accumulateBases();
84 void accumulateVPtrs();
85 void accumulateVBases();
86 void accumulateFields();
88 accumulateBitFields(RecordDecl::field_iterator field,
90
91 mlir::Type getVFPtrType();
92
93 bool isAAPCS() const {
94 return astContext.getTargetInfo().getABI().starts_with("aapcs");
95 }
96
97 /// Helper function to check if the target machine is BigEndian.
98 bool isBigEndian() const { return astContext.getTargetInfo().isBigEndian(); }
99
100 // The Itanium base layout rule allows virtual bases to overlap
101 // other bases, which complicates layout in specific ways.
102 //
103 // Note specifically that the ms_struct attribute doesn't change this.
104 bool isOverlappingVBaseABI() {
105 return !astContext.getTargetInfo().getCXXABI().isMicrosoft();
106 }
107 // Recursively searches all of the bases to find out if a vbase is
108 // not the primary vbase of some base class.
109 bool hasOwnStorage(const CXXRecordDecl *decl, const CXXRecordDecl *query);
110
111 /// The Microsoft bitfield layout rule allocates discrete storage
112 /// units of the field's formal type and only combines adjacent
113 /// fields of the same formal type. We want to emit a layout with
114 /// these discrete storage units instead of combining them into a
115 /// continuous run.
116 bool isDiscreteBitFieldABI() {
117 return astContext.getTargetInfo().getCXXABI().isMicrosoft() ||
118 recordDecl->isMsStruct(astContext);
119 }
120
121 CharUnits bitsToCharUnits(uint64_t bitOffset) {
122 return astContext.toCharUnitsFromBits(bitOffset);
123 }
124
125 void calculateZeroInit();
126
127 CharUnits getSize(mlir::Type Ty) {
128 return CharUnits::fromQuantity(dataLayout.layout.getTypeSize(Ty));
129 }
130 CharUnits getSizeInBits(mlir::Type ty) {
131 return CharUnits::fromQuantity(dataLayout.layout.getTypeSizeInBits(ty));
132 }
133 CharUnits getAlignment(mlir::Type Ty) {
134 return CharUnits::fromQuantity(dataLayout.layout.getTypeABIAlignment(Ty));
135 }
136
137 bool isZeroInitializable(const FieldDecl *fd) {
138 return cirGenTypes.isZeroInitializable(fd->getType());
139 }
140 bool isZeroInitializable(const RecordDecl *rd) {
141 return cirGenTypes.isZeroInitializable(rd);
142 }
143
144 /// Wraps cir::IntType with some implicit arguments.
145 mlir::Type getUIntNType(uint64_t numBits) {
146 unsigned alignedBits = llvm::PowerOf2Ceil(numBits);
147 alignedBits = std::max(8u, alignedBits);
148 return cir::IntType::get(&cirGenTypes.getMLIRContext(), alignedBits,
149 /*isSigned=*/false);
150 }
151
152 mlir::Type getCharType() {
153 return cir::IntType::get(&cirGenTypes.getMLIRContext(),
154 astContext.getCharWidth(),
155 /*isSigned=*/false);
156 }
157
158 mlir::Type getByteArrayType(CharUnits numberOfChars) {
159 assert(!numberOfChars.isZero() && "Empty byte arrays aren't allowed.");
160 mlir::Type type = getCharType();
161 return numberOfChars == CharUnits::One()
162 ? type
163 : cir::ArrayType::get(type, numberOfChars.getQuantity());
164 }
165
166 // Gets the CIR BaseSubobject type from a CXXRecordDecl.
167 mlir::Type getStorageType(const CXXRecordDecl *RD) {
168 return cirGenTypes.getCIRGenRecordLayout(RD).getBaseSubobjectCIRType();
169 }
170 // This is different from LLVM traditional codegen because CIRGen uses arrays
171 // of bytes instead of arbitrary-sized integers. This is important for packed
172 // structures support.
173 mlir::Type getBitfieldStorageType(unsigned numBits) {
174 unsigned alignedBits = llvm::alignTo(numBits, astContext.getCharWidth());
175 if (cir::isValidFundamentalIntWidth(alignedBits))
176 return builder.getUIntNTy(alignedBits);
177
178 mlir::Type type = getCharType();
179 return cir::ArrayType::get(type, alignedBits / astContext.getCharWidth());
180 }
181
182 mlir::Type getStorageType(const FieldDecl *fieldDecl) {
183 mlir::Type type = cirGenTypes.convertTypeForMem(fieldDecl->getType());
184 if (fieldDecl->isBitField()) {
185 cirGenTypes.getCGModule().errorNYI(recordDecl->getSourceRange(),
186 "getStorageType for bitfields");
187 }
188 return type;
189 }
190
191 uint64_t getFieldBitOffset(const FieldDecl *fieldDecl) {
192 return astRecordLayout.getFieldOffset(fieldDecl->getFieldIndex());
193 }
194
195 /// Fills out the structures that are ultimately consumed.
196 void fillOutputFields();
197
198 void appendPaddingBytes(CharUnits size) {
199 if (size.isZero())
200 return;
201 mlir::Type padTy = getByteArrayType(size);
202 padded = true;
203 if (recordDecl->isUnion()) {
204 assert(!unionPadding && "at most one union tail-padding type");
205 unionPadding = padTy;
206 } else {
207 fieldTypes.push_back(padTy);
208 }
209 }
210
211 CIRGenTypes &cirGenTypes;
212 CIRGenBuilderTy &builder;
213 const ASTContext &astContext;
214 const RecordDecl *recordDecl;
215 const CXXRecordDecl *cxxRecordDecl;
216 const ASTRecordLayout &astRecordLayout;
217 // Helpful intermediate data-structures
218 std::vector<MemberInfo> members;
219 // Output fields, consumed by CIRGenTypes::computeRecordLayout
220 llvm::SmallVector<mlir::Type, 16> fieldTypes;
221 mlir::Type unionPadding;
222 llvm::DenseMap<const FieldDecl *, CIRGenBitFieldInfo> bitFields;
223 llvm::DenseMap<const FieldDecl *, unsigned> fieldIdxMap;
224 llvm::DenseMap<const CXXRecordDecl *, unsigned> nonVirtualBases;
225 llvm::DenseMap<const CXXRecordDecl *, unsigned> virtualBases;
226 cir::CIRDataLayout dataLayout;
227
228 LLVM_PREFERRED_TYPE(bool)
229 unsigned zeroInitializable : 1;
230 LLVM_PREFERRED_TYPE(bool)
231 unsigned zeroInitializableAsBase : 1;
232 LLVM_PREFERRED_TYPE(bool)
233 unsigned packed : 1;
234 LLVM_PREFERRED_TYPE(bool)
235 unsigned padded : 1;
236
237private:
238 CIRRecordLowering(const CIRRecordLowering &) = delete;
239 void operator=(const CIRRecordLowering &) = delete;
240}; // CIRRecordLowering
241} // namespace
242
243CIRRecordLowering::CIRRecordLowering(CIRGenTypes &cirGenTypes,
244 const RecordDecl *recordDecl, bool packed)
245 : cirGenTypes{cirGenTypes}, builder{cirGenTypes.getBuilder()},
246 astContext{cirGenTypes.getASTContext()}, recordDecl{recordDecl},
248 astRecordLayout{
249 cirGenTypes.getASTContext().getASTRecordLayout(recordDecl)},
250 dataLayout{cirGenTypes.getCGModule().getModule()},
251 zeroInitializable{true}, zeroInitializableAsBase{true}, packed{packed},
252 padded{false} {}
253
254void CIRRecordLowering::setBitFieldInfo(const FieldDecl *fd,
255 CharUnits startOffset,
256 mlir::Type storageType) {
257 CIRGenBitFieldInfo &info = bitFields[fd->getCanonicalDecl()];
259 info.offset =
260 (unsigned)(getFieldBitOffset(fd) - astContext.toBits(startOffset));
261 info.size = fd->getBitWidthValue();
262 info.storageSize = getSizeInBits(storageType).getQuantity();
263 info.storageOffset = startOffset;
264 info.storageType = storageType;
265 info.name = fd->getName();
266
267 if (info.size > info.storageSize)
268 info.size = info.storageSize;
269 // Reverse the bit offsets for big endian machines. Since bitfields are laid
270 // out as packed bits within an integer-sized unit, we can imagine the bits
271 // counting from the most-significant-bit instead of the
272 // least-significant-bit.
273 if (dataLayout.isBigEndian())
274 info.offset = info.storageSize - (info.offset + info.size);
275
276 info.volatileStorageSize = 0;
277 info.volatileOffset = 0;
279}
280
281void CIRRecordLowering::lower(bool nonVirtualBaseType) {
282 if (recordDecl->isUnion()) {
283 lowerUnion();
284 computeVolatileBitfields();
285 return;
286 }
287
288 CharUnits size = nonVirtualBaseType ? astRecordLayout.getNonVirtualSize()
289 : astRecordLayout.getSize();
290
291 accumulateFields();
292
293 if (cxxRecordDecl) {
294 accumulateVPtrs();
295 accumulateBases();
296 if (members.empty()) {
297 appendPaddingBytes(size);
298 computeVolatileBitfields();
299 return;
300 }
301 if (!nonVirtualBaseType)
302 accumulateVBases();
303 }
304
305 llvm::stable_sort(members);
306 // TODO: Verify bitfield clipping
308
309 members.push_back(makeStorageInfo(size, getUIntNType(8)));
310 determinePacked(nonVirtualBaseType);
311 insertPadding();
312 members.pop_back();
313
314 calculateZeroInit();
315 fillOutputFields();
316 computeVolatileBitfields();
317}
318
319void CIRRecordLowering::fillOutputFields() {
320 for (const MemberInfo &member : members) {
321 if (member.data)
322 fieldTypes.push_back(member.data);
323 if (member.kind == MemberInfo::InfoKind::Field) {
324 if (member.fieldDecl)
325 fieldIdxMap[member.fieldDecl->getCanonicalDecl()] =
326 fieldTypes.size() - 1;
327 // A field without storage must be a bitfield.
328 if (!member.data) {
329 assert(member.fieldDecl &&
330 "member.data is a nullptr so member.fieldDecl should not be");
331 setBitFieldInfo(member.fieldDecl, member.offset, fieldTypes.back());
332 }
333 } else if (member.kind == MemberInfo::InfoKind::Base) {
334 nonVirtualBases[member.cxxRecordDecl] = fieldTypes.size() - 1;
335 } else if (member.kind == MemberInfo::InfoKind::VBase) {
336 virtualBases[member.cxxRecordDecl] = fieldTypes.size() - 1;
337 }
338 }
339}
340
342CIRRecordLowering::accumulateBitFields(RecordDecl::field_iterator field,
344 if (isDiscreteBitFieldABI()) {
345 // run stores the first element of the current run of bitfields. fieldEnd is
346 // used as a special value to note that we don't have a current run. A
347 // bitfield run is a contiguous collection of bitfields that can be stored
348 // in the same storage block. Zero-sized bitfields and bitfields that would
349 // cross an alignment boundary break a run and start a new one.
351 // tail is the offset of the first bit off the end of the current run. It's
352 // used to determine if the ASTRecordLayout is treating these two bitfields
353 // as contiguous. StartBitOffset is offset of the beginning of the Run.
354 uint64_t startBitOffset, tail = 0;
355 for (; field != fieldEnd && field->isBitField(); ++field) {
356 // Zero-width bitfields end runs.
357 if (field->isZeroLengthBitField()) {
358 run = fieldEnd;
359 continue;
360 }
361 uint64_t bitOffset = getFieldBitOffset(*field);
362 mlir::Type type = cirGenTypes.convertTypeForMem(field->getType());
363 // If we don't have a run yet, or don't live within the previous run's
364 // allocated storage then we allocate some storage and start a new run.
365 if (run == fieldEnd || bitOffset >= tail) {
366 run = field;
367 startBitOffset = bitOffset;
368 tail = startBitOffset + dataLayout.getTypeAllocSizeInBits(type);
369 // Add the storage member to the record. This must be added to the
370 // record before the bitfield members so that it gets laid out before
371 // the bitfields it contains get laid out.
372 members.push_back(
373 makeStorageInfo(bitsToCharUnits(startBitOffset), type));
374 }
375 // Bitfields get the offset of their storage but come afterward and remain
376 // there after a stable sort.
377 members.push_back(MemberInfo(bitsToCharUnits(startBitOffset),
378 MemberInfo::InfoKind::Field, nullptr,
379 *field));
380 }
381 return field;
382 }
383
384 CharUnits regSize =
385 bitsToCharUnits(astContext.getTargetInfo().getRegisterWidth());
386 unsigned charBits = astContext.getCharWidth();
387
388 // Data about the start of the span we're accumulating to create an access
389 // unit from. 'Begin' is the first bitfield of the span. If 'begin' is
390 // 'fieldEnd', we've not got a current span. The span starts at the
391 // 'beginOffset' character boundary. 'bitSizeSinceBegin' is the size (in bits)
392 // of the span -- this might include padding when we've advanced to a
393 // subsequent bitfield run.
394 RecordDecl::field_iterator begin = fieldEnd;
395 CharUnits beginOffset;
396 uint64_t bitSizeSinceBegin;
397
398 // The (non-inclusive) end of the largest acceptable access unit we've found
399 // since 'begin'. If this is 'begin', we're gathering the initial set of
400 // bitfields of a new span. 'bestEndOffset' is the end of that acceptable
401 // access unit -- it might extend beyond the last character of the bitfield
402 // run, using available padding characters.
403 RecordDecl::field_iterator bestEnd = begin;
404 CharUnits bestEndOffset;
405 bool bestClipped; // Whether the representation must be in a byte array.
406
407 for (;;) {
408 // atAlignedBoundary is true if 'field' is the (potential) start of a new
409 // span (or the end of the bitfields). When true, limitOffset is the
410 // character offset of that span and barrier indicates whether the new
411 // span cannot be merged into the current one.
412 bool atAlignedBoundary = false;
413 bool barrier = false; // a barrier can be a zero Bit Width or non bit member
414 if (field != fieldEnd && field->isBitField()) {
415 uint64_t bitOffset = getFieldBitOffset(*field);
416 if (begin == fieldEnd) {
417 // Beginning a new span.
418 begin = field;
419 bestEnd = begin;
420
421 assert((bitOffset % charBits) == 0 && "Not at start of char");
422 beginOffset = bitsToCharUnits(bitOffset);
423 bitSizeSinceBegin = 0;
424 } else if ((bitOffset % charBits) != 0) {
425 // Bitfield occupies the same character as previous bitfield, it must be
426 // part of the same span. This can include zero-length bitfields, should
427 // the target not align them to character boundaries. Such non-alignment
428 // is at variance with the standards, which require zero-length
429 // bitfields be a barrier between access units. But of course we can't
430 // achieve that in the middle of a character.
431 assert(bitOffset ==
432 astContext.toBits(beginOffset) + bitSizeSinceBegin &&
433 "Concatenating non-contiguous bitfields");
434 } else {
435 // Bitfield potentially begins a new span. This includes zero-length
436 // bitfields on non-aligning targets that lie at character boundaries
437 // (those are barriers to merging).
438 if (field->isZeroLengthBitField())
439 barrier = true;
440 atAlignedBoundary = true;
441 }
442 } else {
443 // We've reached the end of the bitfield run. Either we're done, or this
444 // is a barrier for the current span.
445 if (begin == fieldEnd)
446 break;
447
448 barrier = true;
449 atAlignedBoundary = true;
450 }
451
452 // 'installBest' indicates whether we should create an access unit for the
453 // current best span: fields ['begin', 'bestEnd') occupying characters
454 // ['beginOffset', 'bestEndOffset').
455 bool installBest = false;
456 if (atAlignedBoundary) {
457 // 'field' is the start of a new span or the end of the bitfields. The
458 // just-seen span now extends to 'bitSizeSinceBegin'.
459
460 // Determine if we can accumulate that just-seen span into the current
461 // accumulation.
462 CharUnits accessSize = bitsToCharUnits(bitSizeSinceBegin + charBits - 1);
463 if (bestEnd == begin) {
464 // This is the initial run at the start of a new span. By definition,
465 // this is the best seen so far.
466 bestEnd = field;
467 bestEndOffset = beginOffset + accessSize;
468 // Assume clipped until proven not below.
469 bestClipped = true;
470 if (!bitSizeSinceBegin)
471 // A zero-sized initial span -- this will install nothing and reset
472 // for another.
473 installBest = true;
474 } else if (accessSize > regSize) {
475 // Accumulating the just-seen span would create a multi-register access
476 // unit, which would increase register pressure.
477 installBest = true;
478 }
479
480 if (!installBest) {
481 // Determine if accumulating the just-seen span will create an expensive
482 // access unit or not.
483 mlir::Type type = getUIntNType(astContext.toBits(accessSize));
485 cirGenTypes.getCGModule().errorNYI(
486 field->getSourceRange(), "NYI CheapUnalignedBitFieldAccess");
487
488 if (!installBest) {
489 // Find the next used storage offset to determine what the limit of
490 // the current span is. That's either the offset of the next field
491 // with storage (which might be field itself) or the end of the
492 // non-reusable tail padding.
493 CharUnits limitOffset;
494 for (auto probe = field; probe != fieldEnd; ++probe)
495 if (!isEmptyFieldForLayout(astContext, *probe)) {
496 // A member with storage sets the limit.
497 assert((getFieldBitOffset(*probe) % charBits) == 0 &&
498 "Next storage is not byte-aligned");
499 limitOffset = bitsToCharUnits(getFieldBitOffset(*probe));
500 goto FoundLimit;
501 }
502 limitOffset = cxxRecordDecl ? astRecordLayout.getNonVirtualSize()
503 : astRecordLayout.getDataSize();
504
505 FoundLimit:
506 CharUnits typeSize = getSize(type);
507 if (beginOffset + typeSize <= limitOffset) {
508 // There is space before limitOffset to create a naturally-sized
509 // access unit.
510 bestEndOffset = beginOffset + typeSize;
511 bestEnd = field;
512 bestClipped = false;
513 }
514 if (barrier) {
515 // The next field is a barrier that we cannot merge across.
516 installBest = true;
517 } else if (cirGenTypes.getCGModule()
519 .FineGrainedBitfieldAccesses) {
520 installBest = true;
521 } else {
522 // Otherwise, we're not installing. Update the bit size
523 // of the current span to go all the way to limitOffset, which is
524 // the (aligned) offset of next bitfield to consider.
525 bitSizeSinceBegin = astContext.toBits(limitOffset - beginOffset);
526 }
527 }
528 }
529 }
530
531 if (installBest) {
532 assert((field == fieldEnd || !field->isBitField() ||
533 (getFieldBitOffset(*field) % charBits) == 0) &&
534 "Installing but not at an aligned bitfield or limit");
535 CharUnits accessSize = bestEndOffset - beginOffset;
536 if (!accessSize.isZero()) {
537 // Add the storage member for the access unit to the record. The
538 // bitfields get the offset of their storage but come afterward and
539 // remain there after a stable sort.
540 mlir::Type type;
541 if (bestClipped) {
542 assert(getSize(getUIntNType(astContext.toBits(accessSize))) >
543 accessSize &&
544 "Clipped access need not be clipped");
545 type = getByteArrayType(accessSize);
546 } else {
547 type = getUIntNType(astContext.toBits(accessSize));
548 assert(getSize(type) == accessSize &&
549 "Unclipped access must be clipped");
550 }
551 members.push_back(makeStorageInfo(beginOffset, type));
552 for (; begin != bestEnd; ++begin)
553 if (!begin->isZeroLengthBitField())
554 members.push_back(MemberInfo(
555 beginOffset, MemberInfo::InfoKind::Field, nullptr, *begin));
556 }
557 // Reset to start a new span.
558 field = bestEnd;
559 begin = fieldEnd;
560 } else {
561 assert(field != fieldEnd && field->isBitField() &&
562 "Accumulating past end of bitfields");
563 assert(!barrier && "Accumulating across barrier");
564 // Accumulate this bitfield into the current (potential) span.
565 bitSizeSinceBegin += field->getBitWidthValue();
566 ++field;
567 }
568 }
569
570 return field;
571}
572
573void CIRRecordLowering::accumulateFields() {
574 for (RecordDecl::field_iterator field = recordDecl->field_begin(),
575 fieldEnd = recordDecl->field_end();
576 field != fieldEnd;) {
577 if (field->isBitField()) {
578 field = accumulateBitFields(field, fieldEnd);
579 assert((field == fieldEnd || !field->isBitField()) &&
580 "Failed to accumulate all the bitfields");
581 } else if (isEmptyFieldForLayout(astContext, *field)) {
582 // TODO(cir): do we want to do anything special about zero size members?
584 ++field;
585 } else {
586 // Use base subobject layout for potentially-overlapping fields,
587 // as it is done in RecordLayoutBuilder.
588 members.push_back(MemberInfo(
589 bitsToCharUnits(getFieldBitOffset(*field)),
590 MemberInfo::InfoKind::Field,
591 field->isPotentiallyOverlapping()
592 ? getStorageType(field->getType()->getAsCXXRecordDecl())
593 : getStorageType(*field),
594 *field));
595 ++field;
596 }
597 }
598}
599
600void CIRRecordLowering::calculateZeroInit() {
601 for (const MemberInfo &member : members) {
602 if (member.kind == MemberInfo::InfoKind::Field) {
603 if (!member.fieldDecl || isZeroInitializable(member.fieldDecl))
604 continue;
605 zeroInitializable = zeroInitializableAsBase = false;
606 return;
607 } else if (member.kind == MemberInfo::InfoKind::Base ||
608 member.kind == MemberInfo::InfoKind::VBase) {
609 if (isZeroInitializable(member.cxxRecordDecl))
610 continue;
611 zeroInitializable = false;
612 if (member.kind == MemberInfo::InfoKind::Base)
613 zeroInitializableAsBase = false;
614 }
615 }
616}
617
618void CIRRecordLowering::determinePacked(bool nvBaseType) {
619 if (packed)
620 return;
621 CharUnits alignment = CharUnits::One();
622 CharUnits nvAlignment = CharUnits::One();
623 CharUnits nvSize = !nvBaseType && cxxRecordDecl
624 ? astRecordLayout.getNonVirtualSize()
625 : CharUnits::Zero();
626
627 for (const MemberInfo &member : members) {
628 if (!member.data)
629 continue;
630 // If any member falls at an offset that it not a multiple of its alignment,
631 // then the entire record must be packed.
632 if (!member.offset.isMultipleOf(getAlignment(member.data)))
633 packed = true;
634 if (member.offset < nvSize)
635 nvAlignment = std::max(nvAlignment, getAlignment(member.data));
636 alignment = std::max(alignment, getAlignment(member.data));
637 }
638 // If the size of the record (the capstone's offset) is not a multiple of the
639 // record's alignment, it must be packed.
640 if (!members.back().offset.isMultipleOf(alignment))
641 packed = true;
642 // If the non-virtual sub-object is not a multiple of the non-virtual
643 // sub-object's alignment, it must be packed. We cannot have a packed
644 // non-virtual sub-object and an unpacked complete object or vise versa.
645 if (!nvSize.isMultipleOf(nvAlignment))
646 packed = true;
647 // Update the alignment of the sentinel.
648 if (!packed)
649 members.back().data = getUIntNType(astContext.toBits(alignment));
650}
651
652void CIRRecordLowering::insertPadding() {
653 std::vector<std::pair<CharUnits, CharUnits>> padding;
654 CharUnits size = CharUnits::Zero();
655 for (const MemberInfo &member : members) {
656 if (!member.data)
657 continue;
658 CharUnits offset = member.offset;
659 assert(offset >= size);
660 // Insert padding if we need to.
661 if (offset !=
662 size.alignTo(packed ? CharUnits::One() : getAlignment(member.data)))
663 padding.push_back(std::make_pair(size, offset - size));
664 size = offset + getSize(member.data);
665 }
666 if (padding.empty())
667 return;
668 padded = true;
669 // Add the padding to the Members list and sort it.
670 for (const std::pair<CharUnits, CharUnits> &paddingPair : padding)
671 members.push_back(makeStorageInfo(paddingPair.first,
672 getByteArrayType(paddingPair.second)));
673 llvm::stable_sort(members);
674}
675
676static cir::ArgPassingKind
678 switch (kind) {
680 return cir::ArgPassingKind::CanPassInRegs;
682 return cir::ArgPassingKind::CannotPassInRegs;
684 return cir::ArgPassingKind::CanNeverPassInRegs;
685 }
686 llvm_unreachable("unknown RecordArgPassingKind");
687}
688
689std::unique_ptr<CIRGenRecordLayout>
691 CIRRecordLowering lowering(*this, rd, /*packed=*/false);
692 assert(ty->isIncomplete() && "recomputing record layout?");
693 lowering.lower(/*nonVirtualBaseType=*/false);
694
695 // If we're in C++, compute the base subobject type. For C++ records the base
696 // subobject type is always set (matching classic CodeGen). For unions and
697 // final classes the base subobject and complete object types are identical
698 // (no tail padding can be reused), so baseTy points at the same record as
699 // ty. We must still populate baseTy in those cases because callers such as
700 // getStorageType(const CXXRecordDecl *) used to lay out potentially-
701 // overlapping ([[no_unique_address]]) fields read it unconditionally; a
702 // null baseTy would otherwise propagate as a null mlir::Type into the
703 // members vector and trip the !empty() assertion in fillOutputFields.
704 cir::RecordType baseTy;
705 if (llvm::isa<CXXRecordDecl>(rd)) {
706 baseTy = *ty;
707 if (!rd->isUnion() && !rd->hasAttr<FinalAttr>() &&
708 lowering.astRecordLayout.getNonVirtualSize() !=
709 lowering.astRecordLayout.getSize()) {
710 CIRRecordLowering baseLowering(*this, rd, /*Packed=*/lowering.packed);
711 baseLowering.lower(/*NonVirtualBaseType=*/true);
712 std::string baseIdentifier = getRecordTypeName(rd, ".base");
713 baseTy = builder.getCompleteNamedRecordType(
714 baseLowering.fieldTypes, baseLowering.packed, baseLowering.padded,
715 baseIdentifier);
716 // TODO(cir): add something like addRecordTypeName
717
718 // BaseTy and Ty must agree on their packedness for getCIRFieldNo to work
719 // on both of them with the same index.
720 assert(lowering.packed == baseLowering.packed &&
721 "Non-virtual and complete types must agree on packedness");
722 }
723 }
724
725 // Fill in the record *after* computing the base type. Filling in the body
726 // signifies that the type is no longer opaque and record layout is complete,
727 // but we may need to recursively layout rd while laying D out as a base type.
729 ty->complete(lowering.fieldTypes, lowering.packed, lowering.padded,
730 lowering.unionPadding);
731
732 // Queue ABI metadata for the module-level cir.record_layouts attribute.
733 if (ty->getName()) {
734 mlir::MLIRContext *mlirCtx = ty->getContext();
735 cir::ArgPassingKind apk =
737
738 bool hasTrivialDestructor = true;
739 if (auto *cxxRD = dyn_cast<CXXRecordDecl>(rd))
740 hasTrivialDestructor = cxxRD->hasTrivialDestructor();
741 const auto &astLayout = astContext.getASTRecordLayout(rd);
742 uint64_t recordAlignInBytes = astLayout.getAlignment().getQuantity();
743
744 cgm.addRecordLayout(ty->getName(), cir::RecordLayoutAttr::get(
745 mlirCtx, apk, hasTrivialDestructor,
746 recordAlignInBytes));
747 }
748
749 auto rl = std::make_unique<CIRGenRecordLayout>(
750 ty ? *ty : cir::RecordType{}, baseTy ? baseTy : cir::RecordType{},
751 (bool)lowering.zeroInitializable, (bool)lowering.zeroInitializableAsBase);
752
753 rl->nonVirtualBases.swap(lowering.nonVirtualBases);
754 rl->completeObjectVirtualBases.swap(lowering.virtualBases);
755
756 // Add all the field numbers.
757 rl->fieldIdxMap.swap(lowering.fieldIdxMap);
758
759 rl->bitFields.swap(lowering.bitFields);
760
761 // Dump the layout, if requested.
762 if (getASTContext().getLangOpts().DumpRecordLayouts) {
763 llvm::outs() << "\n*** Dumping CIRgen Record Layout\n";
764 llvm::outs() << "Record: ";
765 rd->dump(llvm::outs());
766 llvm::outs() << "\nLayout: ";
767 rl->print(llvm::outs());
768 }
769
770 // TODO: implement verification
771 return rl;
772}
773
774void CIRGenRecordLayout::print(raw_ostream &os) const {
775 os << "<CIRecordLayout\n";
776 os << " CIR Type:" << completeObjectType << "\n";
777 if (baseSubobjectType)
778 os << " NonVirtualBaseCIRType:" << baseSubobjectType << "\n";
779 os << " IsZeroInitializable:" << zeroInitializable << "\n";
780 os << " BitFields:[\n";
781 std::vector<std::pair<unsigned, const CIRGenBitFieldInfo *>> bitInfo;
782 for (auto &[decl, info] : bitFields) {
783 const RecordDecl *rd = decl->getParent();
784 unsigned index = 0;
785 for (RecordDecl::field_iterator it = rd->field_begin(); *it != decl; ++it)
786 ++index;
787 bitInfo.push_back(std::make_pair(index, &info));
788 }
789 llvm::array_pod_sort(bitInfo.begin(), bitInfo.end());
790 for (std::pair<unsigned, const CIRGenBitFieldInfo *> &info : bitInfo) {
791 os.indent(4);
792 info.second->print(os);
793 os << "\n";
794 }
795 os << " ]>\n";
796}
797
798void CIRGenBitFieldInfo::print(raw_ostream &os) const {
799 os << "<CIRBitFieldInfo" << " name:" << name << " offset:" << offset
800 << " size:" << size << " isSigned:" << isSigned
801 << " storageSize:" << storageSize
802 << " storageOffset:" << storageOffset.getQuantity()
803 << " volatileOffset:" << volatileOffset
804 << " volatileStorageSize:" << volatileStorageSize
805 << " volatileStorageOffset:" << volatileStorageOffset.getQuantity() << ">";
806}
807
808void CIRGenRecordLayout::dump() const { print(llvm::errs()); }
809
810void CIRGenBitFieldInfo::dump() const { print(llvm::errs()); }
811
812void CIRRecordLowering::lowerUnion() {
813 CharUnits layoutSize = astRecordLayout.getSize();
814 mlir::Type storageType = nullptr;
815 bool seenNamedMember = false;
816
817 // Iterate through the fields setting bitFieldInfo and the Fields array. Also
818 // locate the "most appropriate" storage type.
819 for (const FieldDecl *field : recordDecl->fields()) {
820 mlir::Type fieldType;
821 if (field->isBitField()) {
822 if (field->isZeroLengthBitField())
823 continue;
824 fieldType = getBitfieldStorageType(field->getBitWidthValue());
825 setBitFieldInfo(field, CharUnits::Zero(), fieldType);
826 } else {
827 fieldType = getStorageType(field);
828 }
829
830 // This maps a field to its index. For unions, the index is always 0.
831 fieldIdxMap[field->getCanonicalDecl()] = 0;
832
833 // Compute zero-initializable status.
834 // This union might not be zero initialized: it may contain a pointer to
835 // data member which might have some exotic initialization sequence.
836 // If this is the case, then we ought not to try and come up with a "better"
837 // type, it might not be very easy to come up with a Constant which
838 // correctly initializes it.
839 if (!seenNamedMember) {
840 seenNamedMember = field->getIdentifier();
841 if (!seenNamedMember)
842 if (const RecordDecl *fieldRD = field->getType()->getAsRecordDecl())
843 seenNamedMember = fieldRD->findFirstNamedDataMember();
844 if (seenNamedMember && !isZeroInitializable(field)) {
845 zeroInitializable = zeroInitializableAsBase = false;
846 storageType = fieldType;
847 }
848 }
849
850 // Because our union isn't zero initializable, we won't be getting a better
851 // storage type.
852 if (!zeroInitializable)
853 continue;
854
855 // Conditionally update our storage type if we've got a new "better" one.
856 if (!storageType || getAlignment(fieldType) > getAlignment(storageType) ||
857 (getAlignment(fieldType) == getAlignment(storageType) &&
858 getSize(fieldType) > getSize(storageType)))
859 storageType = fieldType;
860
861 // NOTE(cir): Track all union member's types, not just the largest one. It
862 // allows for proper type-checking and retain more info for analisys.
863 fieldTypes.push_back(fieldType);
864 }
865
866 if (!storageType) {
867 appendPaddingBytes(layoutSize);
868 return;
869 }
870
871 if (layoutSize < getSize(storageType))
872 storageType = getByteArrayType(layoutSize);
873 else
874 appendPaddingBytes(layoutSize - getSize(storageType));
875
876 // Set packed if we need it.
877 if (!layoutSize.isMultipleOf(getAlignment(storageType)))
878 packed = true;
879}
880
881bool CIRRecordLowering::hasOwnStorage(const CXXRecordDecl *decl,
882 const CXXRecordDecl *query) {
883 const ASTRecordLayout &declLayout = astContext.getASTRecordLayout(decl);
884 if (declLayout.isPrimaryBaseVirtual() && declLayout.getPrimaryBase() == query)
885 return false;
886 for (const auto &base : decl->bases())
887 if (!hasOwnStorage(base.getType()->getAsCXXRecordDecl(), query))
888 return false;
889 return true;
890}
891
892/// The AAPCS that defines that, when possible, bit-fields should
893/// be accessed using containers of the declared type width:
894/// When a volatile bit-field is read, and its container does not overlap with
895/// any non-bit-field member or any zero length bit-field member, its container
896/// must be read exactly once using the access width appropriate to the type of
897/// the container. When a volatile bit-field is written, and its container does
898/// not overlap with any non-bit-field member or any zero-length bit-field
899/// member, its container must be read exactly once and written exactly once
900/// using the access width appropriate to the type of the container. The two
901/// accesses are not atomic.
902///
903/// Enforcing the width restriction can be disabled using
904/// -fno-aapcs-bitfield-width.
905void CIRRecordLowering::computeVolatileBitfields() {
906 if (!isAAPCS() ||
907 !cirGenTypes.getCGModule().getCodeGenOpts().AAPCSBitfieldWidth)
908 return;
909
910 for (auto &[field, info] : bitFields) {
911 mlir::Type resLTy = cirGenTypes.convertTypeForMem(field->getType());
912
913 if (astContext.toBits(astRecordLayout.getAlignment()) <
914 getSizeInBits(resLTy).getQuantity())
915 continue;
916
917 // CIRRecordLowering::setBitFieldInfo() pre-adjusts the bit-field offsets
918 // for big-endian targets, but it assumes a container of width
919 // info.storageSize. Since AAPCS uses a different container size (width
920 // of the type), we first undo that calculation here and redo it once
921 // the bit-field offset within the new container is calculated.
922 const unsigned oldOffset =
923 isBigEndian() ? info.storageSize - (info.offset + info.size)
924 : info.offset;
925 // Offset to the bit-field from the beginning of the struct.
926 const unsigned absoluteOffset =
927 astContext.toBits(info.storageOffset) + oldOffset;
928
929 // Container size is the width of the bit-field type.
930 const unsigned storageSize = getSizeInBits(resLTy).getQuantity();
931 // Nothing to do if the access uses the desired
932 // container width and is naturally aligned.
933 if (info.storageSize == storageSize && (oldOffset % storageSize == 0))
934 continue;
935
936 // Offset within the container.
937 unsigned offset = absoluteOffset & (storageSize - 1);
938 // Bail out if an aligned load of the container cannot cover the entire
939 // bit-field. This can happen for example, if the bit-field is part of a
940 // packed struct. AAPCS does not define access rules for such cases, we let
941 // clang to follow its own rules.
942 if (offset + info.size > storageSize)
943 continue;
944
945 // Re-adjust offsets for big-endian targets.
946 if (isBigEndian())
947 offset = storageSize - (offset + info.size);
948
949 const CharUnits storageOffset =
950 astContext.toCharUnitsFromBits(absoluteOffset & ~(storageSize - 1));
951 const CharUnits end = storageOffset +
952 astContext.toCharUnitsFromBits(storageSize) -
954
955 const ASTRecordLayout &layout =
956 astContext.getASTRecordLayout(field->getParent());
957 // If we access outside memory outside the record, than bail out.
958 const CharUnits recordSize = layout.getSize();
959 if (end >= recordSize)
960 continue;
961
962 // Bail out if performing this load would access non-bit-fields members.
963 bool conflict = false;
964 for (const auto *f : recordDecl->fields()) {
965 // Allow sized bit-fields overlaps.
966 if (f->isBitField() && !f->isZeroLengthBitField())
967 continue;
968
969 const CharUnits fOffset = astContext.toCharUnitsFromBits(
970 layout.getFieldOffset(f->getFieldIndex()));
971
972 // As C11 defines, a zero sized bit-field defines a barrier, so
973 // fields after and before it should be race condition free.
974 // The AAPCS acknowledges it and imposes no restritions when the
975 // natural container overlaps a zero-length bit-field.
976 if (f->isZeroLengthBitField()) {
977 if (end > fOffset && storageOffset < fOffset) {
978 conflict = true;
979 break;
980 }
981 }
982
983 const CharUnits fEnd =
984 fOffset +
985 astContext.toCharUnitsFromBits(
986 getSizeInBits(cirGenTypes.convertTypeForMem(f->getType()))
987 .getQuantity()) -
989 // If no overlap, continue.
990 if (end < fOffset || fEnd < storageOffset)
991 continue;
992
993 // The desired load overlaps a non-bit-field member, bail out.
994 conflict = true;
995 break;
996 }
997
998 if (conflict)
999 continue;
1000 // Write the new bit-field access parameters.
1001 // As the storage offset now is defined as the number of elements from the
1002 // start of the structure, we should divide the Offset by the element size.
1004 storageOffset /
1005 astContext.toCharUnitsFromBits(storageSize).getQuantity();
1006 info.volatileStorageSize = storageSize;
1007 info.volatileOffset = offset;
1008 }
1009}
1010
1011void CIRRecordLowering::accumulateBases() {
1012 // If we've got a primary virtual base, we need to add it with the bases.
1013 if (astRecordLayout.isPrimaryBaseVirtual()) {
1014 const CXXRecordDecl *baseDecl = astRecordLayout.getPrimaryBase();
1015 members.push_back(MemberInfo(CharUnits::Zero(), MemberInfo::InfoKind::Base,
1016 getStorageType(baseDecl), baseDecl));
1017 }
1018
1019 // Accumulate the non-virtual bases.
1020 for (const auto &base : cxxRecordDecl->bases()) {
1021 if (base.isVirtual())
1022 continue;
1023 // Bases can be zero-sized even if not technically empty if they
1024 // contain only a trailing array member.
1025 const CXXRecordDecl *baseDecl = base.getType()->getAsCXXRecordDecl();
1026 if (!baseDecl->isEmpty() &&
1027 !astContext.getASTRecordLayout(baseDecl).getNonVirtualSize().isZero()) {
1028 members.push_back(MemberInfo(astRecordLayout.getBaseClassOffset(baseDecl),
1029 MemberInfo::InfoKind::Base,
1030 getStorageType(baseDecl), baseDecl));
1031 }
1032 }
1033}
1034
1035void CIRRecordLowering::accumulateVBases() {
1036 for (const auto &base : cxxRecordDecl->vbases()) {
1037 const CXXRecordDecl *baseDecl = base.getType()->getAsCXXRecordDecl();
1038 if (isEmptyRecordForLayout(astContext, base.getType()))
1039 continue;
1040 CharUnits offset = astRecordLayout.getVBaseClassOffset(baseDecl);
1041 // If the vbase is a primary virtual base of some base, then it doesn't
1042 // get its own storage location but instead lives inside of that base.
1043 if (isOverlappingVBaseABI() && astContext.isNearlyEmpty(baseDecl) &&
1044 !hasOwnStorage(cxxRecordDecl, baseDecl)) {
1045 members.push_back(
1046 MemberInfo(offset, MemberInfo::InfoKind::VBase, nullptr, baseDecl));
1047 continue;
1048 }
1049 // If we've got a vtordisp, add it as a storage type.
1050 if (astRecordLayout.getVBaseOffsetsMap()
1051 .find(baseDecl)
1052 ->second.hasVtorDisp())
1053 members.push_back(makeStorageInfo(offset - CharUnits::fromQuantity(4),
1054 getUIntNType(32)));
1055 members.push_back(MemberInfo(offset, MemberInfo::InfoKind::VBase,
1056 getStorageType(baseDecl), baseDecl));
1057 }
1058}
1059
1060void CIRRecordLowering::accumulateVPtrs() {
1061 if (astRecordLayout.hasOwnVFPtr())
1062 members.push_back(MemberInfo(CharUnits::Zero(), MemberInfo::InfoKind::VFPtr,
1063 getVFPtrType()));
1064
1065 if (astRecordLayout.hasOwnVBPtr())
1066 cirGenTypes.getCGModule().errorNYI(recordDecl->getSourceRange(),
1067 "accumulateVPtrs: hasOwnVBPtr");
1068}
1069
1070mlir::Type CIRRecordLowering::getVFPtrType() {
1071 return cir::VPtrType::get(builder.getContext());
1072}
Defines the clang::ASTContext interface.
static bool isAAPCS(const TargetInfo &targetInfo)
Helper method to check if the underlying ABI is AAPCS.
static cir::ArgPassingKind convertRecordArgPassingKind(RecordArgPassingKind kind)
Defines the C++ Decl subclasses, other than those for templates (found in DeclTemplate....
static void print(llvm::raw_ostream &OS, const T &V, const Context &Ctx, QualType Ty)
*collection of selector each with an associated kind and an ordered *collection of selectors A selector has a kind
bool isBigEndian() const
llvm::TypeSize getTypeAllocSizeInBits(mlir::Type ty) const
Returns the offset in bits between successive objects of the specified type, including alignment padd...
C++ view class that accepts both !cir.struct and !cir.union types.
Definition CIRTypes.h:93
bool isIncomplete() const
Definition CIRTypes.cpp:502
void complete(llvm::ArrayRef< mlir::Type > members, bool packed, bool padded, mlir::Type padding={})
Definition CIRTypes.cpp:535
mlir::StringAttr getName() const
Definition CIRTypes.cpp:497
const ASTRecordLayout & getASTRecordLayout(const RecordDecl *D) const
Get or compute information about the layout of the specified record (struct/union/class) D,...
bool isNearlyEmpty(const CXXRecordDecl *RD) const
int64_t toBits(CharUnits CharSize) const
Convert a size in characters to a size in bits.
const TargetInfo & getTargetInfo() const
Definition ASTContext.h:921
CharUnits toCharUnitsFromBits(int64_t BitSize) const
Convert a size in bits to a size in characters.
uint64_t getCharWidth() const
Return the size of the character type, in bits.
bool hasOwnVFPtr() const
hasOwnVFPtr - Does this class provide its own virtual-function table pointer, rather than inheriting ...
CharUnits getAlignment() const
getAlignment - Get the record alignment in characters.
bool hasOwnVBPtr() const
hasOwnVBPtr - Does this class provide its own virtual-base table pointer, rather than inheriting one ...
CharUnits getSize() const
getSize - Get the record size in characters.
uint64_t getFieldOffset(unsigned FieldNo) const
getFieldOffset - Get the offset of the given field index, in bits.
CharUnits getDataSize() const
getDataSize() - Get the record data size, which is the record size without tail padding,...
CharUnits getBaseClassOffset(const CXXRecordDecl *Base) const
getBaseClassOffset - Get the offset, in chars, for the given base class.
CharUnits getVBaseClassOffset(const CXXRecordDecl *VBase) const
getVBaseClassOffset - Get the offset, in chars, for the given base class.
const VBaseOffsetsMapTy & getVBaseOffsetsMap() const
const CXXRecordDecl * getPrimaryBase() const
getPrimaryBase - Get the primary base for this record.
bool isPrimaryBaseVirtual() const
isPrimaryBaseVirtual - Get whether the primary base for this record is virtual or not.
CharUnits getNonVirtualSize() const
getNonVirtualSize - Get the non-virtual size (in chars) of an object, which is the size of the object...
DiagnosticBuilder errorNYI(SourceLocation, llvm::StringRef)
Helpers to emit "not yet implemented" error diagnostics.
const clang::CodeGenOptions & getCodeGenOpts() const
This class organizes the cross-module state that is used while lowering AST types to CIR types.
Definition CIRGenTypes.h:50
CIRGenModule & getCGModule() const
Definition CIRGenTypes.h:90
std::string getRecordTypeName(const clang::RecordDecl *, llvm::StringRef suffix)
clang::ASTContext & getASTContext() const
std::unique_ptr< CIRGenRecordLayout > computeRecordLayout(const clang::RecordDecl *rd, cir::RecordType *ty)
mlir::Type convertTypeForMem(clang::QualType, bool forBitField=false)
Convert type T into an mlir::Type.
Represents a C++ struct/union/class.
Definition DeclCXX.h:258
bool isEmpty() const
Determine whether this is an empty class in the sense of (C++11 [meta.unary.prop]).
Definition DeclCXX.h:1191
CharUnits - This is an opaque type for sizes expressed in character units.
Definition CharUnits.h:38
bool isZero() const
isZero - Test whether the quantity equals zero.
Definition CharUnits.h:122
QuantityType getQuantity() const
getQuantity - Get the raw integer representation of this quantity.
Definition CharUnits.h:185
static CharUnits One()
One - Construct a CharUnits quantity of one.
Definition CharUnits.h:58
bool isMultipleOf(CharUnits N) const
Test whether this is a multiple of the other value.
Definition CharUnits.h:143
static CharUnits fromQuantity(QuantityType Quantity)
fromQuantity - Construct a CharUnits quantity from a raw integer type.
Definition CharUnits.h:63
CharUnits alignTo(const CharUnits &Align) const
alignTo - Returns the next integer (mod 2**64) that is greater than or equal to this quantity and is ...
Definition CharUnits.h:201
static CharUnits Zero()
Zero - Construct a CharUnits quantity of zero.
Definition CharUnits.h:53
void dump() const
bool hasAttr() const
Definition DeclBase.h:585
Represents a member of a struct/union/class.
Definition Decl.h:3182
unsigned getBitWidthValue() const
Computes the bit width of this field, if this is a bit field.
Definition Decl.cpp:4748
FieldDecl * getCanonicalDecl() override
Retrieves the canonical declaration of this field.
Definition Decl.h:3429
StringRef getName() const
Get the name of identifier for this declaration as a StringRef.
Definition Decl.h:301
Represents a struct/union/class.
Definition Decl.h:4347
RecordArgPassingKind getArgPassingRestrictions() const
Definition Decl.h:4488
specific_decl_iterator< FieldDecl > field_iterator
Definition Decl.h:4547
field_iterator field_begin() const
Definition Decl.cpp:5271
bool isUnion() const
Definition Decl.h:3950
virtual unsigned getRegisterWidth() const
Return the "preferred" register width on this target.
Definition TargetInfo.h:907
bool hasCheapUnalignedBitFieldAccess() const
Return true iff unaligned accesses are cheap.
Definition TargetInfo.h:921
bool isSignedIntegerOrEnumerationType() const
Determines whether this is an integer type that is signed or an enumeration types whose underlying ty...
Definition Type.cpp:2289
QualType getType() const
Definition Decl.h:723
bool isValidFundamentalIntWidth(unsigned width)
Definition CIRTypes.cpp:825
bool isEmptyFieldForLayout(const ASTContext &context, const FieldDecl *fd)
isEmptyFieldForLayout - Return true if the field is "empty", that is, either a zero-width bit-field o...
bool isEmptyRecordForLayout(const ASTContext &context, QualType t)
isEmptyRecordForLayout - Return true if a structure contains only empty base classes (per isEmptyReco...
const internal::VariadicAllOfMatcher< Type > type
Matches Types in the clang AST.
const internal::VariadicDynCastAllOfMatcher< Decl, FieldDecl > fieldDecl
Matches field declarations.
const internal::VariadicDynCastAllOfMatcher< Decl, CXXRecordDecl > cxxRecordDecl
Matches C++ class declarations.
const internal::VariadicAllOfMatcher< Decl > decl
Matches declarations.
const internal::VariadicDynCastAllOfMatcher< Decl, RecordDecl > recordDecl
Matches class, struct, and union declarations.
RangeSelector member(std::string ID)
Given a MemberExpr, selects the member token. ID is the node's binding in the match result.
Stencil run(MatchConsumer< std::string > C)
Wraps a MatchConsumer in a Stencil, so that it can be used in a Stencil.
Definition Stencil.cpp:489
The JSON file list parser is used to communicate input to InstallAPI.
bool operator<(DeclarationName LHS, DeclarationName RHS)
Ordering on two declaration names.
RecordArgPassingKind
Enum that represents the different ways arguments are passed to and returned from function calls.
Definition Decl.h:4324
@ CanPassInRegs
The argument of this type can be passed directly in registers.
Definition Decl.h:4326
@ CanNeverPassInRegs
The argument of this type cannot be passed directly in registers.
Definition Decl.h:4340
@ CannotPassInRegs
The argument of this type cannot be passed directly in registers.
Definition Decl.h:4335
unsigned long uint64_t
Diagnostic wrappers for TextAPI types for error reporting.
Definition Dominators.h:30
void __ovld __conv barrier(cl_mem_fence_flags)
All work-items in a work-group executing the kernel on a processor must execute this function before ...
#define false
Definition stdbool.h:26
#define true
Definition stdbool.h:25
static bool zeroSizeRecordMembers()
static bool checkBitfieldClipping()
static bool astRecordDeclAttr()
unsigned offset
The offset within a contiguous run of bitfields that are represented as a single "field" within the c...
void print(llvm::raw_ostream &os) const
unsigned storageSize
The storage size in bits which should be used when accessing this bitfield.
unsigned volatileStorageSize
The storage size in bits which should be used when accessing this bitfield.
clang::CharUnits storageOffset
The offset of the bitfield storage from the start of the record.
unsigned size
The total size of the bit-field, in bits.
unsigned isSigned
Whether the bit-field is signed.
clang::CharUnits volatileStorageOffset
The offset of the bitfield storage from the start of the record.
unsigned volatileOffset
The offset within a contiguous run of bitfields that are represented as a single "field" within the c...
llvm::StringRef name
The name of a bitfield.