diff options
author | Douglas Gregor <dgregor@apple.com> | 2009-01-28 23:36:17 +0000 |
---|---|---|
committer | Douglas Gregor <dgregor@apple.com> | 2009-01-28 23:36:17 +0000 |
commit | 34e7946831a63f96d3ba3478c74ca8e25ee52d7e (patch) | |
tree | abb077e17e60200375521dcaa14875641f5ad747 /lib/CodeGen | |
parent | 939abced90071beb750041ee9c2cf57f827e024a (diff) |
Improvements to code-generation and semantic analysis of designated
initializers.
- We now initialize unions properly when a member other than the
first is named by a designated initializer.
- We now provide proper semantic analysis and code generation for
GNU array-range designators *except* that side effects will occur
more than once. We warn about this.
git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@63253 91177308-0d34-0410-b5e6-96231b3b80d8
Diffstat (limited to 'lib/CodeGen')
-rw-r--r-- | lib/CodeGen/CGExprAgg.cpp | 15 | ||||
-rw-r--r-- | lib/CodeGen/CGExprConstant.cpp | 12 | ||||
-rw-r--r-- | lib/CodeGen/CodeGenTypes.cpp | 1 |
3 files changed, 22 insertions, 6 deletions
diff --git a/lib/CodeGen/CGExprAgg.cpp b/lib/CodeGen/CGExprAgg.cpp index 2534a14cb6..075a12c57e 100644 --- a/lib/CodeGen/CGExprAgg.cpp +++ b/lib/CodeGen/CGExprAgg.cpp @@ -454,10 +454,17 @@ void AggExprEmitter::VisitInitListExpr(InitListExpr *E) { if (Field->getType()->isIncompleteArrayType()) break; - if (Field->getIdentifier() == 0) { - // Initializers can't initialize unnamed fields, e.g. "int : 20;" + if (Field->isUnnamedBitfield()) continue; - } + + // When we're coping with C99 designated initializers into a + // union, find the field that has the same type as the expression + // we're initializing the union with. + if (isUnion && CurInitVal < NumInitElements && + (CGF.getContext().getCanonicalType(Field->getType()) != + CGF.getContext().getCanonicalType(E->getInit(CurInitVal)->getType()))) + continue; + // FIXME: volatility LValue FieldLoc = CGF.EmitLValueForField(DestPtr, *Field, isUnion,0); if (CurInitVal < NumInitElements) { @@ -471,8 +478,6 @@ void AggExprEmitter::VisitInitListExpr(InitListExpr *E) { } // Unions only initialize one field. - // (FIXME: things can get weird with designators, but they aren't - // supported yet.) if (isUnion) break; } diff --git a/lib/CodeGen/CGExprConstant.cpp b/lib/CodeGen/CGExprConstant.cpp index 87868ed347..b05048c946 100644 --- a/lib/CodeGen/CGExprConstant.cpp +++ b/lib/CodeGen/CGExprConstant.cpp @@ -246,12 +246,22 @@ public: // first field int FieldNo = 0; // Field no in RecordDecl FieldDecl* curField = 0; + bool sawAnyFields = false; for (RecordDecl::field_iterator Field = RD->field_begin(), FieldEnd = RD->field_end(); Field != FieldEnd; ++Field) { curField = *Field; FieldNo++; - if (curField->getIdentifier()) + + if (curField->isUnnamedBitfield()) + continue; + + // If we have an initializer, find the field whose type is the + // same as that initializer. This + sawAnyFields = true; + if (ILE->getNumInits() > 0 && + CGM.getContext().getCanonicalType(curField->getType()) == + CGM.getContext().getCanonicalType(ILE->getInit(0)->getType())) break; } diff --git a/lib/CodeGen/CodeGenTypes.cpp b/lib/CodeGen/CodeGenTypes.cpp index aac6e8d644..edd16087fd 100644 --- a/lib/CodeGen/CodeGenTypes.cpp +++ b/lib/CodeGen/CodeGenTypes.cpp @@ -518,6 +518,7 @@ void RecordOrganizer::layoutUnionFields(const ASTRecordLayout &RL) { Field != FieldEnd; ++Field) { // The offset should usually be zero, but bitfields could be strange uint64_t offset = RL.getFieldOffset(curField); + CGT.ConvertTypeRecursive(Field->getType()); if (Field->isBitField()) { Expr *BitWidth = Field->getBitWidth(); |