2013-03-26 09:15:19 +08:00
// RUN: %clang_cc1 -fsyntax-only -verify -std=c++98 %s
// RUN: %clang_cc1 -fsyntax-only -verify -std=c++11 %s
2010-01-14 01:31:36 +08:00
struct X0 {
2017-01-20 05:00:13 +08:00
X0 ( ) ;
X0 ( int ) ;
2010-01-14 01:31:36 +08:00
X0 f1 ( ) ;
X0 f2 ( ) ;
2017-01-20 05:00:13 +08:00
typedef int A ;
typedef X0 B ;
2010-01-14 01:31:36 +08:00
} ;
template < typename T >
2017-01-20 05:00:13 +08:00
struct X1 : X0 {
X1 ( ) ;
2010-01-14 01:31:36 +08:00
X1 < T > ( int ) ;
( X1 < T > ) ( float ) ;
X1 f2 ( ) ;
X1 f2 ( int ) ;
X1 f2 ( float ) ;
2017-01-20 05:00:13 +08:00
X1 f2 ( double ) ;
2017-02-02 05:36:38 +08:00
X1 f2 ( short ) ;
X1 f2 ( long ) ;
2010-01-14 01:31:36 +08:00
} ;
// Error recovery: out-of-line constructors whose names have template arguments.
template < typename T > X1 < T > : : X1 < T > ( int ) { } // expected-error{{out-of-line constructor for 'X1' cannot have template arguments}}
template < typename T > ( X1 < T > : : X1 < T > ) ( float ) { } // expected-error{{out-of-line constructor for 'X1' cannot have template arguments}}
// Error recovery: out-of-line constructor names intended to be types
2017-01-20 05:00:13 +08:00
X0 : : X0 X0 : : f1 ( ) { return X0 ( ) ; } // expected-error{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
2010-01-14 01:31:36 +08:00
struct X0 : : X0 X0 : : f2 ( ) { return X0 ( ) ; }
2017-02-02 05:36:38 +08:00
template < typename T > X1 < T > : : X1 < T > X1 < T > : : f2 ( ) { } // expected-error{{missing 'typename'}}
template < typename T > X1 < T > : : X1 < T > ( X1 < T > : : f2 ) ( int ) { } // expected-error{{missing 'typename'}}
2010-01-14 01:31:36 +08:00
template < typename T > struct X1 < T > : : X1 < T > ( X1 < T > : : f2 ) ( float ) { }
2017-01-20 05:00:13 +08:00
template < typename T > struct X1 < T > : : X1 ( X1 < T > : : f2 ) ( double ) { }
2017-02-02 05:36:38 +08:00
template < typename T > typename X1 < T > : : template X1 < T > X1 < T > : : f2 ( short ) { } // expected-warning {{qualified reference to 'X1' is a constructor name rather than a template name in this context}}
template < typename T > typename X1 < T > : : template X1 < T > ( X1 < T > : : f2 ) ( long ) { } // expected-warning {{qualified reference to 'X1' is a constructor name rather than a template name in this context}}
2017-01-20 05:00:13 +08:00
void x1test ( X1 < int > x1i ) {
x1i . f2 ( ) ;
x1i . f2 ( 0 ) ;
x1i . f2 ( 0.f ) ;
x1i . f2 ( 0. ) ;
}
void other_contexts ( ) {
X0 : : X0 x0 ; // expected-error{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
X1 < int > : : X1 x1a ; // expected-error{{qualified reference to 'X1' is a constructor name rather than a type in this context}}
X1 < int > : : X1 < float > x1b ; // expected-error{{qualified reference to 'X1' is a constructor name rather than a template name in this context}}
X0 : : B ok1 ;
X0 : : X0 : : A ok2 ;
X0 : : X0 : : X0 x0b ; // expected-error{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
X1 < int > : : X0 ok3 ;
X1 < int > : : X0 : : X0 x0c ; // expected-error{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
X1 < int > : : X1 < float > : : X0 ok4 ;
{
typename X0 : : X0 tn1 ; // expected-warning{{qualified reference to 'X0' is a constructor name rather than a type in this context}} expected-warning 0-1{{typename}}
typename X1 < int > : : X1 < float > tn2 ; // expected-warning{{qualified reference to 'X1' is a constructor name rather than a template name in this context}} expected-warning 0-1{{typename}}
typename X0 : : B ok1 ; // expected-warning 0-1{{typename}}
typename X1 < int > : : X0 ok2 ; // expected-warning 0-1{{typename}}
}
{
struct X0 : : X0 tag1 ;
struct X1 < int > : : X1 tag2 ;
struct X1 < int > : : X1 < int > tag3 ;
}
int a ;
{
X0 : : X0 ( a ) ; // expected-error{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
}
}
template < typename T > void in_instantiation_x0 ( ) {
typename T : : X0 x0 ; // expected-warning{{qualified reference to 'X0' is a constructor name rather than a type in this context}}
typename T : : A a ;
typename T : : B b ;
}
template void in_instantiation_x0 < X0 > ( ) ; // expected-note {{instantiation of}}
template < typename T > void in_instantiation_x1 ( ) {
typename T : : X1 x1 ; // expected-warning{{qualified reference to 'X1' is a constructor name rather than a type in this context}}
2017-01-20 08:20:39 +08:00
typename T : : template X1 < int > x1i ; // expected-warning{{qualified reference to 'X1' is a constructor name rather than a template name in this context}}
2017-01-20 05:00:13 +08:00
typename T : : X0 x0 ;
}
template void in_instantiation_x1 < X1 < int > > ( ) ; // expected-note {{instantiation of}}
namespace sfinae {
template < typename T > void f ( typename T : : X0 * ) = delete ; // expected-warning 0-1{{extension}}
template < typename T > void f ( . . . ) ;
void g ( ) { f < X0 > ( 0 ) ; }
}
namespace versus_injected_class_name {
template < typename T > struct A : T : : B {
struct T : : B * p ;
typename T : : B : : type a ;
A ( ) : T : : B ( ) { }
typename T : : B b ; // expected-warning {{qualified reference to 'B' is a constructor name rather than a type in this context}}
} ;
struct B {
typedef int type ;
} ;
template struct A < B > ; // expected-note {{in instantiation of}}
}
2013-03-26 09:15:19 +08:00
// We have a special case for lookup within using-declarations that are
// member-declarations: foo::bar::baz::baz always names baz's constructor
// in such a context, even if looking up 'baz' within foo::bar::baz would
// not find the injected-class-name. Likewise foo::bar::baz<T>::baz also
// names the constructor.
namespace InhCtor {
struct A {
A ( int ) ;
protected :
int T ( ) ;
} ;
typedef A T ;
struct B : A {
// This is a using-declaration for 'int A::T()' in C++98, but is an
// inheriting constructor declaration in C++11.
using InhCtor : : T : : T ;
} ;
# if __cplusplus < 201103L
B b ( 123 ) ; // expected-error {{no matching constructor}}
// expected-note@-7 2{{candidate constructor}}
int n = b . T ( ) ; // ok, accessible
# else
B b ( 123 ) ; // ok, inheriting constructor
int n = b . T ( ) ; // expected-error {{'T' is a protected member of 'InhCtor::A'}}
// expected-note@-15 {{declared protected here}}
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
// FIXME: EDG and GCC reject this too, but it's not clear why it would be
// ill-formed.
2013-03-26 09:15:19 +08:00
template < typename T >
struct S : T {
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
struct U : S { // expected-note 6{{candidate}}
2013-03-26 09:15:19 +08:00
using S : : S ;
} ;
using T : : T ;
} ;
P0136R1, DR1573, DR1645, DR1715, DR1736, DR1903, DR1941, DR1959, DR1991:
Replace inheriting constructors implementation with new approach, voted into
C++ last year as a DR against C++11.
Instead of synthesizing a set of derived class constructors for each inherited
base class constructor, we make the constructors of the base class visible to
constructor lookup in the derived class, using the normal rules for
using-declarations.
For constructors, UsingShadowDecl now has a ConstructorUsingShadowDecl derived
class that tracks the requisite additional information. We create shadow
constructors (not found by name lookup) in the derived class to model the
actual initialization, and have a new expression node,
CXXInheritedCtorInitExpr, to model the initialization of a base class from such
a constructor. (This initialization is special because it performs real perfect
forwarding of arguments.)
In cases where argument forwarding is not possible (for inalloca calls,
variadic calls, and calls with callee parameter cleanup), the shadow inheriting
constructor is not emitted and instead we directly emit the initialization code
into the caller of the inherited constructor.
Note that this new model is not perfectly compatible with the old model in some
corner cases. In particular:
* if B inherits a private constructor from A, and C uses that constructor to
construct a B, then we previously required that A befriends B and B
befriends C, but the new rules require A to befriend C directly, and
* if a derived class has its own constructors (and so its implicit default
constructor is suppressed), it may still inherit a default constructor from
a base class
llvm-svn: 274049
2016-06-29 03:03:57 +08:00
S < A > : : U ua ( 0 ) ; // expected-error {{no match}}
S < B > : : U ub ( 0 ) ; // expected-error {{no match}}
2013-03-26 09:15:19 +08:00
template < typename T >
struct X : T {
using T : : Z : : U : : U ;
} ;
template < typename T >
struct X2 : T {
using T : : Z : : template V < int > : : V ;
} ;
struct Y {
struct Z {
typedef Y U ;
template < typename T > using V = Y ;
} ;
Y ( int ) ;
} ;
X < Y > xy ( 0 ) ;
namespace Repeat {
struct A {
struct T {
T ( int ) ;
} ;
} ;
struct Z : A {
using A : : A : : A ;
} ;
template < typename T >
struct ZT : T : : T {
using T : : T : : T ;
} ;
}
namespace NS {
struct NS { } ;
}
struct DerivedFromNS : NS : : NS {
// No special case unless the NNS names a class.
using InhCtor : : NS : : NS ; // expected-error {{using declaration in class refers into 'InhCtor::NS::', which is not a class}}
} ;
2014-02-09 14:54:23 +08:00
// FIXME: Consider reusing the same diagnostic between dependent and non-dependent contexts
2013-03-26 09:15:19 +08:00
typedef int I ;
struct UsingInt {
2015-01-19 03:05:48 +08:00
using I : : I ; // expected-error {{'I' (aka 'int') is not a class, namespace, or enumeration}}
2013-03-26 09:15:19 +08:00
} ;
template < typename T > struct UsingIntTemplate {
using T : : T ; // expected-error {{type 'int' cannot be used prior to '::' because it has no members}}
} ;
UsingIntTemplate < int > uit ; // expected-note {{here}}
# endif
}