]> granicus.if.org Git - clang/commitdiff
Fix for PR16570: when comparing two function pointers, discard qualifiers when
authorRichard Trieu <rtrieu@google.com>
Tue, 6 Aug 2013 03:44:10 +0000 (03:44 +0000)
committerRichard Trieu <rtrieu@google.com>
Tue, 6 Aug 2013 03:44:10 +0000 (03:44 +0000)
comparing non-reference function parameters.  The qualifiers don't matter for
comparisons.

git-svn-id: https://llvm.org/svn/llvm-project/cfe/trunk@187769 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Sema/SemaOverload.cpp
test/SemaCXX/function-pointer-arguments.cpp [new file with mode: 0644]

index 1a9b82962ba6d77b5bdc91b2c1ff8d8fc3332048..fae4d0232153280ddccd184ffed72c75291fa6da 100644 (file)
@@ -2584,9 +2584,17 @@ bool Sema::FunctionArgTypesAreEqual(const FunctionProtoType *OldType,
   for (FunctionProtoType::arg_type_iterator O = OldType->arg_type_begin(),
        N = NewType->arg_type_begin(),
        E = OldType->arg_type_end(); O && (O != E); ++O, ++N) {
-    if (!Context.hasSameType(*O, *N)) {
-      if (ArgPos) *ArgPos = O - OldType->arg_type_begin();
-      return false;
+    if (!(*O)->isReferenceType() && !(*N)->isReferenceType()) {
+      if (!Context.hasSameType(O->getUnqualifiedType(),
+                               N->getUnqualifiedType())) {
+        if (ArgPos) *ArgPos = O - OldType->arg_type_begin();
+        return false;
+      }
+    } else {
+      if (!Context.hasSameType(*O, *N)) {
+        if (ArgPos) *ArgPos = O - OldType->arg_type_begin();
+        return false;
+      }
     }
   }
   return true;
diff --git a/test/SemaCXX/function-pointer-arguments.cpp b/test/SemaCXX/function-pointer-arguments.cpp
new file mode 100644 (file)
index 0000000..9f3fb0a
--- /dev/null
@@ -0,0 +1,52 @@
+//RUN: %clang_cc1 -fsyntax-only -verify %s
+
+namespace PR16570 {
+  int f1(int, int);
+  int f2(const int, int);
+  int f3(int&, int);
+  int f4(const int&, int);
+
+  void good() {
+    int(*g1)(int, int) = f1;
+    int(*g2)(const int, int) = f1;
+    int(*g3)(volatile int, int) = f1;
+    int(*g4)(int, int) = f2;
+    int(*g5)(const int, int) = f2;
+    int(*g6)(volatile int, int) = f2;
+    int(*g7)(int&, int) = f3;
+    int(*g8)(const int&, int) = f4;
+  }
+
+  void bad() {
+    void (*g1)(int, int) = f1;
+    // expected-error@-1 {{different return type ('void' vs 'int'}}
+    const int (*g2)(int, int) = f1;
+    // expected-error@-1 {{different return type ('const int' vs 'int')}}
+
+    int (*g3)(char, int) = f1;
+    // expected-error@-1 {{type mismatch at 1st parameter ('char' vs 'int')}}
+    int (*g4)(int, char) = f1;
+    // expected-error@-1 {{type mismatch at 2nd parameter ('char' vs 'int')}}
+
+    int (*g5)(int) = f1;
+    // expected-error@-1 {{different number of parameters (1 vs 2)}}
+
+    int (*g6)(int, int, int) = f1;
+    // expected-error@-1 {{different number of parameters (3 vs 2)}}
+
+    int (*g7)(const int, char) = f1;
+    // expected-error@-1 {{type mismatch at 2nd parameter ('char' vs 'int')}}
+    int (*g8)(int, char) = f2;
+    // expected-error@-1 {{type mismatch at 2nd parameter ('char' vs 'int')}}
+    int (*g9)(const int&, char) = f3;
+    // expected-error@-1 {{type mismatch at 1st parameter ('const int &' vs 'int &')}}
+    int (*g10)(int&, char) = f4;
+    // expected-error@-1 {{type mismatch at 1st parameter ('int &' vs 'const int &')}}
+  }
+
+  typedef void (*F)(const char * __restrict__, int);
+  void g(const char *, unsigned);
+  F f = g;
+  // expected-error@-1 {{type mismatch at 2nd parameter ('int' vs 'unsigned int')}}
+
+}