/***/
-/// isEmptyRecord - Return true iff a structure has no non-empty
-/// members. Note that a structure with a flexible array member is not
+static bool isEmptyRecord(ASTContext &Context, QualType T);
+
+/// isEmptyField - Return true iff a the field is "empty", that is it
+/// is an unnamed bit-field or an (array of) empty record(s).
+static bool isEmptyField(ASTContext &Context, const FieldDecl *FD) {
+ if (FD->isUnnamedBitfield())
+ return true;
+
+ QualType FT = FD->getType();
+ // Arrays of empty records count as empty.
+ if (const ConstantArrayType *AT = Context.getAsConstantArrayType(FT))
+ if (isEmptyRecord(Context, AT->getElementType()))
+ return true;
+
+ return isEmptyRecord(Context, FT);
+}
+
+/// isEmptyRecord - Return true iff a structure contains only empty
+/// fields. Note that a structure with a flexible array member is not
/// considered empty.
static bool isEmptyRecord(ASTContext &Context, QualType T) {
const RecordType *RT = T->getAsRecordType();
if (RD->hasFlexibleArrayMember())
return false;
for (RecordDecl::field_iterator i = RD->field_begin(Context),
- e = RD->field_end(Context); i != e; ++i) {
- const FieldDecl *FD = *i;
- if (!isEmptyRecord(Context, FD->getType()))
+ e = RD->field_end(Context); i != e; ++i)
+ if (!isEmptyField(Context, *i))
return false;
- }
return true;
}
const FieldDecl *FD = *i;
QualType FT = FD->getType();
+ // Ignore empty fields.
+ if (isEmptyField(Context, FD))
+ continue;
+
// Treat single element arrays as the element
if (const ConstantArrayType *AT = Context.getAsConstantArrayType(FT))
if (AT->getSize().getZExtValue() == 1)
FT = AT->getElementType();
- // Ignore empty records and padding bit-fields.
- if (isEmptyRecord(Context, FT) || FD->isUnnamedBitfield())
- continue;
-
if (Found)
return 0;
e = RT->getDecl()->field_end(Context); i != e; ++i) {
const FieldDecl *FD = *i;
- // Empty structures are ignored.
- if (isEmptyRecord(Context, FD->getType()))
+ // Empty fields are ignored.
+ if (isEmptyField(Context, FD))
continue;
- // As are arrays of empty structures, but not generally, so we
- // can't add this test higher in this routine.
- if (const ConstantArrayType *AT =
- Context.getAsConstantArrayType(FD->getType()))
- if (isEmptyRecord(Context, AT->getElementType()))
- continue;
-
// Check fields recursively.
if (!shouldReturnTypeInRegister(FD->getType(), Context))
return false;