// ::= Q # const pointer
// ::= R # volatile pointer
// ::= S # const volatile pointer
- if (T->isPointerType()) {
+ if (T->isAnyPointerType() || T->isMemberPointerType()) {
if (!Quals.hasVolatile()) {
Out << 'Q';
} else {
// in there.
mangleQualifiers(Quals, false);
}
- else if (T->isPointerType()) {
+ else if (T->isAnyPointerType() || T->isMemberPointerType()) {
Out << 'P';
}
switch (T->getTypeClass()) {
mangleType(ElementTy.getLocalUnqualifiedType());
}
+// <type> ::= <pointer-to-member-type>
+// <pointer-to-member-type> ::= <pointer-cvr-qualifiers> <cvr-qualifiers>
+// <class name> <type>
void MicrosoftCXXNameMangler::mangleType(const MemberPointerType *T) {
- assert(false && "Don't know how to mangle MemberPointerTypes yet!");
+ QualType PointeeType = T->getPointeeType();
+ if (const FunctionProtoType *FPT = dyn_cast<FunctionProtoType>(PointeeType)) {
+ Out << '8';
+ mangleName(cast<RecordType>(T->getClass())->getDecl());
+ mangleType(FPT, NULL, false, true);
+ } else {
+ mangleQualifiers(PointeeType.getQualifiers(), true);
+ mangleName(cast<RecordType>(T->getClass())->getDecl());
+ mangleType(PointeeType.getLocalUnqualifiedType());
+ }
}
void MicrosoftCXXNameMangler::mangleType(const TemplateTypeParmType *T) {
// CHECK: @"\01?h@@3QAHA"
// CHECK: @"\01?i@@3PAY0BD@HA"
// CHECK: @"\01?j@@3P6GHCE@ZA"
+// CHECK: @"\01?k@@3PTfoo@@DA"
+// CHECK: @"\01?l@@3P8foo@@AAHH@ZA"
int a;
int (__stdcall *j)(signed char, unsigned char);
+const volatile char foo::*k;
+
+int (foo::*l)(int);
+
// Static functions are mangled, too.
// Also make sure calling conventions, arglists, and throw specs work.
static void __stdcall alpha(float a, double b) throw() {}