2014-12-17 09:08:39 +08:00
// RUN: %clang_cc1 -std=c++98 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++11 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++14 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
2018-06-28 04:30:34 +08:00
// RUN: %clang_cc1 -std=c++17 -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
// RUN: %clang_cc1 -std=c++2a -triple x86_64-unknown-unknown %s -verify -fexceptions -fcxx-exceptions -pedantic-errors
2014-12-17 09:08:39 +08:00
2018-04-06 02:55:37 +08:00
# if __cplusplus < 201103L
// expected-error@+1 {{variadic macro}}
# define static_assert(...) __extension__ _Static_assert(__VA_ARGS__)
# endif
2018-07-18 06:24:09 +08:00
# if __cplusplus >= 201103L
namespace std {
typedef decltype ( sizeof ( int ) ) size_t ;
template < typename E > class initializer_list {
const E * begin ;
size_t size ;
public :
initializer_list ( ) ;
} ;
} // std
# endif
C++ DR1611, 1658, 2180: implement "potentially constructed subobject" rules for special member functions.
Essentially, as a base class constructor does not construct virtual bases, such
a constructor for an abstract class does not need the corresponding base class
construction to be valid, and likewise for destructors.
This creates an awkward situation: clang will sometimes generate references to
the complete object and deleting destructors for an abstract class (it puts
them in the construction vtable for a derived class). But we can't generate a
"correct" version of these because we can't generate references to base class
constructors any more (if they're template specializations, say, we might not
have instantiated them and can't assume any other TU will emit a copy).
Fortunately, we don't need to, since no correct program can ever invoke them,
so instead emit symbols that just trap.
We should stop emitting references to these symbols, but still need to emit
definitions for compatibility.
llvm-svn: 296275
2017-02-26 07:53:05 +08:00
namespace dr1611 { // dr1611: dup 1658
struct A { A ( int ) ; } ;
struct B : virtual A { virtual void f ( ) = 0 ; } ;
struct C : B { C ( ) : A ( 0 ) { } void f ( ) ; } ;
C c ;
}
2014-12-17 09:08:39 +08:00
namespace dr1684 { // dr1684: 3.6
# if __cplusplus >= 201103L
struct NonLiteral { // expected-note {{because}}
NonLiteral ( ) ;
constexpr int f ( ) { return 0 ; } // expected-warning 0-1{{will not be implicitly 'const'}}
} ;
constexpr int f ( NonLiteral & ) { return 0 ; }
constexpr int f ( NonLiteral ) { return 0 ; } // expected-error {{not a literal type}}
# endif
}
2015-01-28 09:01:21 +08:00
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
namespace dr1631 { // dr1631: 3.7
2015-01-28 09:01:21 +08:00
# if __cplusplus >= 201103L
// Incorrect overload resolution for single-element initializer-list
struct A { int a [ 1 ] ; } ;
struct B { B ( int ) ; } ;
void f ( B , int ) ;
void f ( B , int , int = 0 ) ;
void f ( int , A ) ;
void test ( ) {
2015-02-12 09:50:05 +08:00
f ( { 0 } , { { 1 } } ) ; // expected-warning {{braces around scalar init}}
2015-01-28 09:01:21 +08:00
}
namespace with_error {
void f ( B , int ) ; // TODO: expected- note {{candidate function}}
void f ( int , A ) ; // expected-note {{candidate function}}
void f ( int , A , int = 0 ) ; // expected-note {{candidate function}}
void test ( ) {
f ( { 0 } , { { 1 } } ) ; // expected-error{{call to 'f' is ambiguous}}
}
}
# endif
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
}
2016-12-10 03:11:50 +08:00
namespace dr1638 { // dr1638: yes
# if __cplusplus >= 201103L
template < typename T > struct A {
enum class E ; // expected-note {{previous}}
enum class F : T ; // expected-note 2{{previous}}
} ;
template < > enum class A < int > : : E ;
template < > enum class A < int > : : E { } ;
template < > enum class A < int > : : F : int ;
template < > enum class A < int > : : F : int { } ;
template < > enum class A < short > : : E : int ;
template < > enum class A < short > : : E : int { } ;
template < > enum class A < short > : : F ; // expected-error {{different underlying type}}
template < > enum class A < char > : : E : char ; // expected-error {{different underlying type}}
template < > enum class A < char > : : F : int ; // expected-error {{different underlying type}}
enum class A < unsigned > : : E ; // expected-error {{template specialization requires 'template<>'}} expected-error {{nested name specifier}}
template enum class A < unsigned > : : E ; // expected-error {{enumerations cannot be explicitly instantiated}}
enum class A < unsigned > : : E * e ; // expected-error {{must use 'enum' not 'enum class'}}
struct B {
friend enum class A < unsigned > : : E ; // expected-error {{must use 'enum' not 'enum class'}}
} ;
# endif
}
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
namespace dr1645 { // dr1645: 3.9
# if __cplusplus >= 201103L
2017-01-14 04:46:54 +08:00
struct A {
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
constexpr A ( int , float = 0 ) ; // expected-note 2{{candidate}}
explicit A ( int , int = 0 ) ; // expected-note 2{{candidate}}
A ( int , int , int = 0 ) = delete ; // expected-note {{candidate}}
} ;
struct B : A { // expected-note 2{{candidate}}
2017-01-14 04:46:54 +08:00
using A : : A ; // expected-note 5{{inherited 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
} ;
constexpr B a ( 0 ) ; // expected-error {{ambiguous}}
constexpr B b ( 0 , 0 ) ; // expected-error {{ambiguous}}
# endif
}
2016-12-10 03:11:50 +08:00
2017-01-13 08:57:54 +08:00
namespace dr1653 { // dr1653: 4 c++17
2016-12-10 03:11:50 +08:00
void f ( bool b ) {
+ + b ;
b + + ;
# if __cplusplus <= 201402L
// expected-warning@-3 {{deprecated}} expected-warning@-2 {{deprecated}}
# else
// expected-error@-5 {{incrementing expression of type bool}} expected-error@-4 {{incrementing expression of type bool}}
# endif
- - b ; // expected-error {{cannot decrement expression of type bool}}
b - - ; // expected-error {{cannot decrement expression of type bool}}
b + = 1 ; // ok
b - = 1 ; // ok
}
}
C++ DR1611, 1658, 2180: implement "potentially constructed subobject" rules for special member functions.
Essentially, as a base class constructor does not construct virtual bases, such
a constructor for an abstract class does not need the corresponding base class
construction to be valid, and likewise for destructors.
This creates an awkward situation: clang will sometimes generate references to
the complete object and deleting destructors for an abstract class (it puts
them in the construction vtable for a derived class). But we can't generate a
"correct" version of these because we can't generate references to base class
constructors any more (if they're template specializations, say, we might not
have instantiated them and can't assume any other TU will emit a copy).
Fortunately, we don't need to, since no correct program can ever invoke them,
so instead emit symbols that just trap.
We should stop emitting references to these symbols, but still need to emit
definitions for compatibility.
llvm-svn: 296275
2017-02-26 07:53:05 +08:00
namespace dr1658 { // dr1658: 5
namespace DefCtor {
class A { A ( ) ; } ; // expected-note 0-2{{here}}
class B { ~ B ( ) ; } ; // expected-note 0-2{{here}}
// The stars align! An abstract class does not construct its virtual bases.
struct C : virtual A { C ( ) ; virtual void foo ( ) = 0 ; } ;
C : : C ( ) = default ; // ok, not deleted, expected-error 0-1{{extension}}
struct D : virtual B { D ( ) ; virtual void foo ( ) = 0 ; } ;
D : : D ( ) = default ; // ok, not deleted, expected-error 0-1{{extension}}
// In all other cases, we are not so lucky.
struct E : A { E ( ) ; virtual void foo ( ) = 0 ; } ;
# if __cplusplus < 201103L
E : : E ( ) = default ; // expected-error {{private default constructor}} expected-error {{extension}} expected-note {{here}}
# else
E : : E ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible default constructor}}
# endif
struct F : virtual A { F ( ) ; } ;
# if __cplusplus < 201103L
F : : F ( ) = default ; // expected-error {{private default constructor}} expected-error {{extension}} expected-note {{here}}
# else
F : : F ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible default constructor}}
# endif
struct G : B { G ( ) ; virtual void foo ( ) = 0 ; } ;
# if __cplusplus < 201103L
G : : G ( ) = default ; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}}
# else
G : : G ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}}
# endif
struct H : virtual B { H ( ) ; } ;
# if __cplusplus < 201103L
H : : H ( ) = default ; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}}
# else
H : : H ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}}
# endif
}
namespace Dtor {
class B { ~ B ( ) ; } ; // expected-note 0-2{{here}}
struct D : virtual B { ~ D ( ) ; virtual void foo ( ) = 0 ; } ;
D : : ~ D ( ) = default ; // ok, not deleted, expected-error 0-1{{extension}}
struct G : B { ~ G ( ) ; virtual void foo ( ) = 0 ; } ;
# if __cplusplus < 201103L
G : : ~ G ( ) = default ; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}}
# else
G : : ~ G ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}}
# endif
struct H : virtual B { ~ H ( ) ; } ;
# if __cplusplus < 201103L
H : : ~ H ( ) = default ; // expected-error@-2 {{private destructor}} expected-error {{extension}} expected-note {{here}}
# else
H : : ~ H ( ) = default ; // expected-error {{would delete}} expected-note@-4{{inaccessible destructor}}
# endif
}
namespace MemInit {
struct A { A ( int ) ; } ; // expected-note {{here}}
struct B : virtual A {
B ( ) { }
virtual void f ( ) = 0 ;
} ;
struct C : virtual A {
C ( ) { } // expected-error {{must explicitly initialize}}
} ;
}
namespace CopyCtorParamType {
struct A { A ( A & ) ; } ;
struct B : virtual A { virtual void f ( ) = 0 ; } ;
struct C : virtual A { virtual void f ( ) ; } ;
struct D : A { virtual void f ( ) = 0 ; } ;
struct X {
friend B : : B ( const B & ) throw ( ) ;
friend C : : C ( C & ) ;
friend D : : D ( D & ) ;
} ;
}
namespace CopyCtor {
class A { A ( const A & ) ; A ( A & & ) ; } ; // expected-note 0-4{{here}} expected-error 0-1{{extension}}
struct C : virtual A { C ( const C & ) ; C ( C & & ) ; virtual void foo ( ) = 0 ; } ; // expected-error 0-1{{extension}}
C : : C ( const C & ) = default ; // expected-error 0-1{{extension}}
C : : C ( C & & ) = default ; // expected-error 0-2{{extension}}
struct E : A { E ( const E & ) ; E ( E & & ) ; virtual void foo ( ) = 0 ; } ; // expected-error 0-1{{extension}}
# if __cplusplus < 201103L
E : : E ( const E & ) = default ; // expected-error {{private copy constructor}} expected-error {{extension}} expected-note {{here}}
E : : E ( E & & ) = default ; // expected-error {{private move constructor}} expected-error 2{{extension}} expected-note {{here}}
# else
E : : E ( const E & ) = default ; // expected-error {{would delete}} expected-note@-5{{inaccessible copy constructor}}
E : : E ( E & & ) = default ; // expected-error {{would delete}} expected-note@-6{{inaccessible move constructor}}
# endif
struct F : virtual A { F ( const F & ) ; F ( F & & ) ; } ; // expected-error 0-1{{extension}}
# if __cplusplus < 201103L
F : : F ( const F & ) = default ; // expected-error {{private copy constructor}} expected-error {{extension}} expected-note {{here}}
F : : F ( F & & ) = default ; // expected-error {{private move constructor}} expected-error 2{{extension}} expected-note {{here}}
# else
F : : F ( const F & ) = default ; // expected-error {{would delete}} expected-note@-5{{inaccessible copy constructor}}
F : : F ( F & & ) = default ; // expected-error {{would delete}} expected-note@-6{{inaccessible move constructor}}
# endif
}
// assignment case is superseded by dr2180
}
2018-04-06 02:55:37 +08:00
namespace dr1672 { // dr1672: 7
struct Empty { } ;
struct A : Empty { } ;
struct B { Empty e ; } ;
struct C : A { B b ; int n ; } ;
struct D : A { int n ; B b ; } ;
static_assert ( ! __is_standard_layout ( C ) , " " ) ;
static_assert ( __is_standard_layout ( D ) , " " ) ;
struct E { B b ; int n ; } ;
struct F { int n ; B b ; } ;
union G { B b ; int n ; } ;
union H { int n ; B b ; } ;
struct X { } ;
template < typename T > struct Y : X , A { T t ; } ;
static_assert ( ! __is_standard_layout ( Y < E > ) , " " ) ;
static_assert ( __is_standard_layout ( Y < F > ) , " " ) ;
static_assert ( ! __is_standard_layout ( Y < G > ) , " " ) ;
static_assert ( ! __is_standard_layout ( Y < H > ) , " " ) ;
static_assert ( ! __is_standard_layout ( Y < X > ) , " " ) ;
}
2018-06-28 04:30:34 +08:00
namespace dr1687 { // dr1687: 7
template < typename T > struct To {
operator T ( ) ; // expected-note 2{{first operand was implicitly converted to type 'int *'}}
// expected-note@-1 {{second operand was implicitly converted to type 'double'}}
# if __cplusplus > 201703L
// expected-note@-3 2{{operand was implicitly converted to type 'dr1687::E}}
# endif
} ;
int * a = To < int * > ( ) + 100.0 ; // expected-error {{invalid operands to binary expression ('To<int *>' and 'double')}}
int * b = To < int * > ( ) + To < double > ( ) ; // expected-error {{invalid operands to binary expression ('To<int *>' and 'To<double>')}}
# if __cplusplus > 201703L
enum E1 { } ;
enum E2 { } ;
auto c = To < E1 > ( ) < = > To < E2 > ( ) ; // expected-error {{invalid operands to binary expression ('To<dr1687::E1>' and 'To<dr1687::E2>')}}
# endif
}
2018-07-18 06:24:09 +08:00
namespace dr1696 { // dr1696: 7
namespace std_examples {
# if __cplusplus >= 201402L
extern struct A a ;
struct A {
const A & x = { A { a , a } } ;
const A & y = { A { } } ; // expected-error {{default member initializer for 'y' needed within definition of enclosing class 'A' outside of member functions}} expected-note {{here}}
} ;
A a { a , a } ;
# endif
}
struct A { A ( ) ; ~ A ( ) ; } ;
# if __cplusplus >= 201103L
struct B {
A & & a ; // expected-note {{declared here}}
B ( ) : a { } { } // expected-error {{reference member 'a' binds to a temporary object whose lifetime would be shorter than the constructed object}}
} b ;
# endif
struct C {
C ( ) ;
const A & a ; // expected-note {{declared here}}
} ;
C : : C ( ) : a ( A ( ) ) { } // expected-error {{reference member 'a' binds to a temporary object whose lifetime would be shorter than the constructed object}}
# if __cplusplus >= 201103L
// This is OK in C++14 onwards, per DR1815, though we don't support that yet:
// D1 d1 = {};
// is equivalent to
// D1 d1 = {A()};
// ... which lifetime-extends the A temporary.
struct D1 {
const A & a = A ( ) ;
# if __cplusplus < 201402L
// expected-error@-2 {{binds to a temporary}}
// expected-note@-4 {{here}}
# else
// expected-warning-re@-5 {{sorry, lifetime extension {{.*}} not supported}}
# endif
} ;
D1 d1 = { } ; // expected-note {{here}}
struct D2 {
const A & a = A ( ) ; // expected-error {{binds to a temporary}}
D2 ( ) { } // expected-note {{used here}}
} ;
struct D3 { // expected-note {{used here}}
const A & a = A ( ) ; // expected-error {{binds to a temporary}}
} ;
D3 d3 ; // expected-note {{first required here}}
struct haslist1 {
std : : initializer_list < int > il ; // expected-note {{'std::initializer_list' member}}
haslist1 ( int i ) : il { i , 2 , 3 } { } // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}}
} ;
struct haslist2 {
std : : initializer_list < int > il ; // expected-note {{'std::initializer_list' member}}
haslist2 ( ) ;
} ;
haslist2 : : haslist2 ( ) : il { 1 , 2 } { } // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}}
struct haslist3 {
std : : initializer_list < int > il = { 1 , 2 , 3 } ;
} ;
struct haslist4 { // expected-note {{in default member initializer}}
std : : initializer_list < int > il = { 1 , 2 , 3 } ; // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}}
} ;
haslist4 hl4 ; // expected-note {{in implicit default constructor}}
struct haslist5 {
std : : initializer_list < int > il = { 1 , 2 , 3 } ; // expected-error {{backing array for 'std::initializer_list' member 'il' is a temporary object}}
haslist5 ( ) { } // expected-note {{in default member initializer}}
} ;
# endif
}