diff options
| author | Argyrios Kyrtzidis <akyrtzi@gmail.com> | 2010-12-10 00:11:00 +0000 |
|---|---|---|
| committer | Argyrios Kyrtzidis <akyrtzi@gmail.com> | 2010-12-10 00:11:00 +0000 |
| commit | 648fcbef5b99878277c8cb1c799074392ba4b596 (patch) | |
| tree | 61820458c15e2bc272ab83d34856fd260688dcc6 /clang/lib/CodeGen | |
| parent | 40fa9b6765aae62675666a1efcf47e0ebb6f9253 (diff) | |
| download | bcm5719-llvm-648fcbef5b99878277c8cb1c799074392ba4b596.tar.gz bcm5719-llvm-648fcbef5b99878277c8cb1c799074392ba4b596.zip | |
Fix another obscure corner layout case.
llvm-svn: 121436
Diffstat (limited to 'clang/lib/CodeGen')
| -rw-r--r-- | clang/lib/CodeGen/CGRecordLayoutBuilder.cpp | 22 |
1 files changed, 13 insertions, 9 deletions
diff --git a/clang/lib/CodeGen/CGRecordLayoutBuilder.cpp b/clang/lib/CodeGen/CGRecordLayoutBuilder.cpp index 4beaa82e5d6..5571e43f785 100644 --- a/clang/lib/CodeGen/CGRecordLayoutBuilder.cpp +++ b/clang/lib/CodeGen/CGRecordLayoutBuilder.cpp @@ -125,7 +125,7 @@ private: const ASTRecordLayout &Layout); /// ComputeNonVirtualBaseType - Compute the non-virtual base field types. - void ComputeNonVirtualBaseType(const CXXRecordDecl *RD); + bool ComputeNonVirtualBaseType(const CXXRecordDecl *RD); /// LayoutField - layout a single field. Returns false if the operation failed /// because the current struct is not packed. @@ -612,7 +612,7 @@ CGRecordLayoutBuilder::LayoutNonVirtualBases(const CXXRecordDecl *RD, } } -void +bool CGRecordLayoutBuilder::ComputeNonVirtualBaseType(const CXXRecordDecl *RD) { const ASTRecordLayout &Layout = Types.getContext().getASTRecordLayout(RD); @@ -624,26 +624,27 @@ CGRecordLayoutBuilder::ComputeNonVirtualBaseType(const CXXRecordDecl *RD) { // First check if we can use the same fields as for the complete class. if (AlignedNonVirtualTypeSize == Layout.getSize() / 8) { NonVirtualBaseTypeIsSameAsCompleteType = true; - return; + return true; } - NonVirtualBaseFieldTypes = FieldTypes; - // Check if we need padding. uint64_t AlignedNextFieldOffset = llvm::RoundUpToAlignment(NextFieldOffsetInBytes, getAlignmentAsLLVMStruct()); - assert(AlignedNextFieldOffset <= AlignedNonVirtualTypeSize && - "Size mismatch!"); + if (AlignedNextFieldOffset > AlignedNonVirtualTypeSize) + return false; // Needs packing. + + NonVirtualBaseFieldTypes = FieldTypes; if (AlignedNonVirtualTypeSize == AlignedNextFieldOffset) { // We don't need any padding. - return; + return true; } uint64_t NumBytes = AlignedNonVirtualTypeSize - AlignedNextFieldOffset; NonVirtualBaseFieldTypes.push_back(getByteArrayType(NumBytes)); + return true; } bool CGRecordLayoutBuilder::LayoutFields(const RecordDecl *D) { @@ -670,7 +671,10 @@ bool CGRecordLayoutBuilder::LayoutFields(const RecordDecl *D) { if (RD) { // We've laid out the non-virtual bases and the fields, now compute the // non-virtual base field types. - ComputeNonVirtualBaseType(RD); + if (!ComputeNonVirtualBaseType(RD)) { + assert(!Packed && "Could not layout even with a packed LLVM struct!"); + return false; + } // And lay out the virtual bases. RD->getIndirectPrimaryBases(IndirectPrimaryBases); |

