bool getInvalidType() const { return InvalidType; }
};
+/// FieldDeclarator - This little struct is used to capture information about
+/// structure field declarators, which is basically just a bitfield size.
+struct FieldDeclarator {
+ Declarator D;
+ Action::ExprTy *BitfieldSize;
+ FieldDeclarator(DeclSpec &DS) : D(DS, Declarator::MemberContext) {
+ BitfieldSize = 0;
+ }
+};
+
+
} // end namespace clang
#endif
class DeclSpec;
class ObjCDeclSpec;
class Declarator;
+ class FieldDeclarator;
class AttributeList;
class Scope;
void ParseStructUnionSpecifier(DeclSpec &DS);
void ParseStructUnionBody(SourceLocation StartLoc, unsigned TagType,
DeclTy *TagDecl);
- void ParseStructDeclaration(DeclTy *TagDecl,
- llvm::SmallVectorImpl<DeclTy*> &FieldDecls);
+ void ParseStructDeclaration(DeclSpec &DS,
+ llvm::SmallVectorImpl<FieldDeclarator> &Fields);
bool isDeclarationSpecifier() const;
bool isTypeSpecifierQualifier() const;
/// declarator[opt] ':' constant-expression
/// [GNU] declarator[opt] ':' constant-expression attributes[opt]
///
-void Parser::ParseStructDeclaration(DeclTy *TagDecl,
- llvm::SmallVectorImpl<DeclTy*> &FieldDecls) {
+void Parser::
+ParseStructDeclaration(DeclSpec &DS,
+ llvm::SmallVectorImpl<FieldDeclarator> &Fields) {
// FIXME: When __extension__ is specified, disable extension diagnostics.
- if (Tok.is(tok::kw___extension__))
+ while (Tok.is(tok::kw___extension__))
ConsumeToken();
// Parse the common specifier-qualifiers-list piece.
- DeclSpec DS;
SourceLocation DSStart = Tok.getLocation();
ParseSpecifierQualifierList(DS);
// TODO: Does specifier-qualifier list correctly check that *something* is
}
// Read struct-declarators until we find the semicolon.
- Declarator DeclaratorInfo(DS, Declarator::MemberContext);
-
+ Fields.push_back(DS);
while (1) {
+ FieldDeclarator &DeclaratorInfo = Fields.back();
+
/// struct-declarator: declarator
/// struct-declarator: declarator[opt] ':' constant-expression
if (Tok.isNot(tok::colon))
- ParseDeclarator(DeclaratorInfo);
+ ParseDeclarator(DeclaratorInfo.D);
- ExprTy *BitfieldSize = 0;
if (Tok.is(tok::colon)) {
ConsumeToken();
ExprResult Res = ParseConstantExpression();
if (Res.isInvalid)
SkipUntil(tok::semi, true, true);
else
- BitfieldSize = Res.Val;
+ DeclaratorInfo.BitfieldSize = Res.Val;
}
// If attributes exist after the declarator, parse them.
if (Tok.is(tok::kw___attribute))
- DeclaratorInfo.AddAttributes(ParseAttributes());
-
- // Install the declarator into the current TagDecl.
- DeclTy *Field = Actions.ActOnField(CurScope, TagDecl,
- DS.getSourceRange().getBegin(),
- DeclaratorInfo, BitfieldSize);
- FieldDecls.push_back(Field);
+ DeclaratorInfo.D.AddAttributes(ParseAttributes());
// If we don't have a comma, it is either the end of the list (a ';')
// or an error, bail out.
ConsumeToken();
// Parse the next declarator.
- DeclaratorInfo.clear();
+ Fields.push_back(DS);
// Attributes are only allowed on the second declarator.
if (Tok.is(tok::kw___attribute))
- DeclaratorInfo.AddAttributes(ParseAttributes());
+ Fields.back().D.AddAttributes(ParseAttributes());
}
}
DeclSpec::getSpecifierName((DeclSpec::TST)TagType));
llvm::SmallVector<DeclTy*, 32> FieldDecls;
-
+ llvm::SmallVector<FieldDeclarator, 8> FieldDeclarators;
+
// While we still have something to read, read the declarations in the struct.
while (Tok.isNot(tok::r_brace) && Tok.isNot(tok::eof)) {
// Each iteration of this loop reads one struct-declaration.
ConsumeToken();
continue;
}
- ParseStructDeclaration(TagDecl, FieldDecls);
+
+ // Parse all the comma separated declarators.
+ DeclSpec DS;
+ FieldDeclarators.clear();
+ ParseStructDeclaration(DS, FieldDeclarators);
+
+ // Convert them all to fields.
+ for (unsigned i = 0, e = FieldDeclarators.size(); i != e; ++i) {
+ FieldDeclarator &FD = FieldDeclarators[i];
+ // Install the declarator into the current TagDecl.
+ DeclTy *Field = Actions.ActOnField(CurScope, TagDecl,
+ DS.getSourceRange().getBegin(),
+ FD.D, FD.BitfieldSize);
+ FieldDecls.push_back(Field);
+ }
+
if (Tok.is(tok::semi)) {
ConsumeToken();
SourceLocation AtLoc) {
assert(Tok.isObjCAtKeyword(tok::objc_property) &&
"ParseObjCPropertyDecl(): Expected @property");
- ObjCDeclSpec DS;
+ ObjCDeclSpec OCDS;
ConsumeToken(); // the "property" identifier
// Parse property attribute list, if any.
if (Tok.is(tok::l_paren)) {
// property has attribute list.
- ParseObjCPropertyAttribute(DS);
+ ParseObjCPropertyAttribute(OCDS);
}
// Parse declaration portion of @property.
llvm::SmallVector<DeclTy*, 8> PropertyDecls;
- ParseStructDeclaration(interfaceDecl, PropertyDecls);
+
+ // Parse all the comma separated declarators.
+ DeclSpec DS;
+ llvm::SmallVector<FieldDeclarator, 8> FieldDeclarators;
+ ParseStructDeclaration(DS, FieldDeclarators);
+
+ // Convert them all to fields.
+ for (unsigned i = 0, e = FieldDeclarators.size(); i != e; ++i) {
+ FieldDeclarator &FD = FieldDeclarators[i];
+ // Install the declarator into interfaceDecl.
+ DeclTy *Field = Actions.ActOnField(CurScope, interfaceDecl,
+ DS.getSourceRange().getBegin(),
+ FD.D, FD.BitfieldSize);
+ PropertyDecls.push_back(Field);
+ }
+
if (Tok.is(tok::semi))
ConsumeToken();
else {
SkipUntil(tok::r_brace, true, true);
}
return Actions.ActOnAddObjCProperties(AtLoc, &PropertyDecls[0],
- PropertyDecls.size(), DS);
+ PropertyDecls.size(), OCDS);
}
/// objc-method-proto:
void Parser::ParseObjCClassInstanceVariables(DeclTy *interfaceDecl,
SourceLocation atLoc) {
assert(Tok.is(tok::l_brace) && "expected {");
- llvm::SmallVector<DeclTy*, 16> IvarDecls;
llvm::SmallVector<DeclTy*, 32> AllIvarDecls;
llvm::SmallVector<tok::ObjCKeywordKind, 32> AllVisibilities;
-
+ llvm::SmallVector<FieldDeclarator, 8> FieldDeclarators;
+
SourceLocation LBraceLoc = ConsumeBrace(); // the "{"
tok::ObjCKeywordKind visibility = tok::objc_private;
ConsumeToken();
continue;
}
+
// Set the default visibility to private.
if (Tok.is(tok::at)) { // parse objc-visibility-spec
ConsumeToken(); // eat the @ sign
continue;
default:
Diag(Tok, diag::err_objc_illegal_visibility_spec);
- ConsumeToken();
continue;
}
}
- ParseStructDeclaration(interfaceDecl, IvarDecls);
- for (unsigned i = 0; i < IvarDecls.size(); i++) {
- AllIvarDecls.push_back(IvarDecls[i]);
+
+ // Parse all the comma separated declarators.
+ DeclSpec DS;
+ FieldDeclarators.clear();
+ ParseStructDeclaration(DS, FieldDeclarators);
+
+ // Convert them all to fields.
+ for (unsigned i = 0, e = FieldDeclarators.size(); i != e; ++i) {
+ FieldDeclarator &FD = FieldDeclarators[i];
+ // Install the declarator into interfaceDecl.
+ DeclTy *Field = Actions.ActOnField(CurScope, interfaceDecl,
+ DS.getSourceRange().getBegin(),
+ FD.D, FD.BitfieldSize);
+ AllIvarDecls.push_back(Field);
AllVisibilities.push_back(visibility);
}
- IvarDecls.clear();
if (Tok.is(tok::semi)) {
ConsumeToken();