The following was found by a customer and is accepted by the other primary
C++ compilers, but fails to compile in Clang:
namespace sss {
double foo(int, double);
template <class T>
T foo(T); // note: target of using declaration
} // namespace sss
namespace oad {
void foo();
}
namespace oad {
using ::sss::foo;
}
namespace sss {
using oad::foo; // note: using declaration
}
namespace sss {
double foo(int, double) { return 0; }
template <class T>
T foo(T t) { // error: declaration conflicts with target of using
return t;
}
} // namespace sss
I believe the issue is that MergeFunctionDecl() was calling
checkUsingShadowRedecl() but only considering a FunctionDecl as a
possible shadow and not FunctionTemplateDecl. The changes in this patch
largely mirror how variable declarations were being handled by also
catching FunctionTemplateDecl.
55 lines
2.0 KiB
C++
55 lines
2.0 KiB
C++
// RUN: %clang_cc1 -fmodules-ts -verify -std=c++17 %s
|
|
// RUN: %clang_cc1 -fmodules-ts -verify -std=c++17 %s -DEXPORT
|
|
// RUN: %clang_cc1 -fmodules-ts -verify -std=c++17 %s -DUSING
|
|
|
|
#ifndef NO_GLOBAL
|
|
extern int var; // expected-note {{previous declaration is here}}
|
|
int func(); // expected-note {{previous declaration is here}}
|
|
struct str; // expected-note {{previous declaration is here}}
|
|
using type = int;
|
|
|
|
template<typename> extern int var_tpl; // expected-note {{previous declaration is here}}
|
|
template<typename> int func_tpl(); // expected-note {{previous declaration is here}}
|
|
template<typename> struct str_tpl; // expected-note {{previous declaration is here}}
|
|
template<typename> using type_tpl = int; // expected-note {{previous declaration is here}}
|
|
|
|
typedef int type;
|
|
namespace ns { using ::func; }
|
|
namespace ns_alias = ns;
|
|
#endif
|
|
|
|
export module M;
|
|
|
|
#ifdef USING
|
|
using ::var;
|
|
using ::func;
|
|
using ::str;
|
|
using ::type;
|
|
using ::var_tpl;
|
|
using ::func_tpl;
|
|
using ::str_tpl;
|
|
using ::type_tpl;
|
|
#endif
|
|
|
|
#ifdef EXPORT
|
|
export {
|
|
#endif
|
|
|
|
extern int var; // expected-error {{declaration of 'var' in module M follows declaration in the global module}}
|
|
int func(); // expected-error {{declaration of 'func' in module M follows declaration in the global module}}
|
|
struct str; // expected-error {{declaration of 'str' in module M follows declaration in the global module}}
|
|
using type = int;
|
|
|
|
template<typename> extern int var_tpl; // expected-error {{declaration of 'var_tpl' in module M follows declaration in the global module}}
|
|
template<typename> int func_tpl(); // expected-error {{declaration of 'func_tpl' in module M follows declaration in the global module}}
|
|
template<typename> struct str_tpl; // expected-error {{declaration of 'str_tpl' in module M follows declaration in the global module}}
|
|
template<typename> using type_tpl = int; // expected-error {{declaration of 'type_tpl' in module M follows declaration in the global module}}
|
|
|
|
typedef int type;
|
|
namespace ns { using ::func; }
|
|
namespace ns_alias = ns;
|
|
|
|
#ifdef EXPORT
|
|
}
|
|
#endif
|