aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFariborz Jahanian <fjahanian@apple.com>2007-10-11 18:08:47 +0000
committerFariborz Jahanian <fjahanian@apple.com>2007-10-11 18:08:47 +0000
commitdfbcce2b55ca8b45ce9d9c09c32e43c3bd73ac2a (patch)
tree7bbeccd10a6e08630e936d8f656a2ff1e78e9241
parent88e0052dd80aa822bf4e27741a4a4f900518cdde (diff)
Implemented parsing of objctive-c protocol conforming type used in
an identifier statement. Fixed up pretty priting to print this type correctly. git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@42866 91177308-0d34-0410-b5e6-96231b3b80d8
-rw-r--r--AST/Type.cpp12
-rw-r--r--Parse/ParseStmt.cpp10
-rw-r--r--test/Parser/objc-type-printing.m19
3 files changed, 37 insertions, 4 deletions
diff --git a/AST/Type.cpp b/AST/Type.cpp
index 7deee66070..af07d7a1be 100644
--- a/AST/Type.cpp
+++ b/AST/Type.cpp
@@ -858,14 +858,18 @@ void ObjcInterfaceType::getAsStringInternal(std::string &InnerString) const {
void ObjcQualifiedInterfaceType::getAsStringInternal(
std::string &InnerString) const {
- InnerString = getInterfaceType()->getDecl()->getName() + '<';
+ if (!InnerString.empty()) // Prefix the basic type, e.g. 'typedefname X'.
+ InnerString = ' ' + InnerString;
+ std::string ObjcQIString = getInterfaceType()->getDecl()->getName();
+ ObjcQIString += '<';
int num = getNumProtocols();
for (int i = 0; i < num; i++) {
- InnerString += getProtocols(i)->getName();
+ ObjcQIString += getProtocols(i)->getName();
if (i < num-1)
- InnerString += ',';
+ ObjcQIString += ',';
}
- InnerString += '>';
+ ObjcQIString += '>';
+ InnerString = ObjcQIString + InnerString;
}
void TagType::getAsStringInternal(std::string &InnerString) const {
diff --git a/Parse/ParseStmt.cpp b/Parse/ParseStmt.cpp
index ad6f3932f6..9df20b4ade 100644
--- a/Parse/ParseStmt.cpp
+++ b/Parse/ParseStmt.cpp
@@ -235,6 +235,16 @@ Parser::StmtResult Parser::ParseIdentifierStatement(bool OnlyStatement) {
IdentTok.getLocation(), PrevSpec,
TypeRep);
assert(!isInvalid && "First declspec can't be invalid!");
+ if (Tok.is(tok::less)) {
+ llvm::SmallVector<IdentifierInfo *, 8> ProtocolRefs;
+ ParseObjCProtocolReferences(ProtocolRefs);
+ llvm::SmallVector<DeclTy *, 8> *ProtocolDecl =
+ new llvm::SmallVector<DeclTy *, 8>;
+ DS.setProtocolQualifiers(ProtocolDecl);
+ Actions.FindProtocolDeclaration(IdentTok.getLocation(),
+ &ProtocolRefs[0], ProtocolRefs.size(),
+ *ProtocolDecl);
+ }
// ParseDeclarationSpecifiers will continue from there.
ParseDeclarationSpecifiers(DS);
diff --git a/test/Parser/objc-type-printing.m b/test/Parser/objc-type-printing.m
new file mode 100644
index 0000000000..5d3cbd994f
--- /dev/null
+++ b/test/Parser/objc-type-printing.m
@@ -0,0 +1,19 @@
+// RUN: clang -ast-print %s
+
+@protocol P1 @end
+@protocol P2 @end
+@protocol P3 @end
+
+@interface INTF
+- (INTF<P1>*) METH;
+@end
+
+void foo()
+{
+ INTF *pintf;
+ INTF<P1>* p1;
+ INTF<P1, P1>* p2;
+ INTF<P1, P3>* p3;
+ INTF<P1, P3, P2>* p4;
+ INTF<P2,P2, P3, P1>* p5;
+}