Skip to content

Commit

Permalink
#2548. Add tests for out parameter in a contrvariant position (#2555)
Browse files Browse the repository at this point in the history
Add tests for out parameter in a contravariant position.
  • Loading branch information
sgrekhov authored Feb 23, 2024
1 parent 7b84042 commit 6dbe0dc
Show file tree
Hide file tree
Showing 6 changed files with 488 additions and 0 deletions.
88 changes: 88 additions & 0 deletions TypeSystem/type-variance/out_non_covariant_A01_t01.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,88 @@
// Copyright (c) 2024, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

/// @assertion Let `D` be the declaration of a class or mixin, enum or a type
/// extension and let `X` be a type parameter declared by `D`.
///
/// If `X` has the variance modifier `out` then it is a compile-time error for
/// `X` to occur in a non-covariant position in a member signature in the body
/// of `D`, except that it is not an error if it occurs in a covariant position
/// in the type annotation of a formal parameter which is covariant (this is a
/// contravariant position in the member signature as a whole).
///
/// In particular, `X` can not be the type of a method parameter (unless it is
/// covariant). It can never be the bound of a type parameter of a generic
/// method
///
/// @description Check that it is a compile-time error if a type parameter has a
/// variance modifier `out` and occurs in a contravariant position in a member
/// signature in the body of `D`
/// @author sgrekhov22@gmail.com
// SharedOptions=--enable-experiment=variance,inline-class

class C<out X> {
C(X x) {} // No error here
factory C.f(X x) => C(x); // And here

void f(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified

void set s(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

mixin M<out X> {
void f(X x);
// ^
// [analyzer] unspecified
// [cfe] unspecified

void set s(X x);
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

enum E<out X> {
e;

const E();
const E.n(X x); // No error here

void f(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified

void set s(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

extension type ET<out X>(X x) {
ET.n(X x) : this.x = x; // No error here

void f(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified

void set s(X x) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

void main() {
print(C);
print(M);
print(E);
print(ET);
}
68 changes: 68 additions & 0 deletions TypeSystem/type-variance/out_non_covariant_A01_t02.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
// Copyright (c) 2024, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

/// @assertion Let `D` be the declaration of a class or mixin, enum or a type
/// extension and let `X` be a type parameter declared by `D`.
///
/// If `X` has the variance modifier `out` then it is a compile-time error for
/// `X` to occur in a non-covariant position in a member signature in the body
/// of `D`, except that it is not an error if it occurs in a covariant position
/// in the type annotation of a formal parameter which is covariant (this is a
/// contravariant position in the member signature as a whole).
///
/// In particular, `X` can not be the type of a method parameter (unless it is
/// covariant). It can never be the bound of a type parameter of a generic
/// method
///
/// @description Check that it is a compile-time error if a type parameter has a
/// variance modifier `out` and occurs in a contravariant position in a member
/// signature in the body of `D`
/// @author sgrekhov22@gmail.com
// SharedOptions=--enable-experiment=variance,inline-class

class C<out X> {
C([X? x]) {} // No error here
factory C.f([X? x]) => C(x); // And here

void f([X? x]) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

mixin M<out X> {
void f([X? x]);
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

enum E<out X> {
e;

const E();
const E.n([X? x]); // No error here

void f([X? x]) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

extension type ET<out X>(X? x) {
ET.n([X? x]) : this.x = x; // No error here

void f([X? x]) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

void main() {
print(C);
print(M);
print(E);
print(ET);
}
68 changes: 68 additions & 0 deletions TypeSystem/type-variance/out_non_covariant_A01_t03.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,68 @@
// Copyright (c) 2024, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

/// @assertion Let `D` be the declaration of a class or mixin, enum or a type
/// extension and let `X` be a type parameter declared by `D`.
///
/// If `X` has the variance modifier `out` then it is a compile-time error for
/// `X` to occur in a non-covariant position in a member signature in the body
/// of `D`, except that it is not an error if it occurs in a covariant position
/// in the type annotation of a formal parameter which is covariant (this is a
/// contravariant position in the member signature as a whole).
///
/// In particular, `X` can not be the type of a method parameter (unless it is
/// covariant). It can never be the bound of a type parameter of a generic
/// method
///
/// @description Check that it is a compile-time error if a type parameter has a
/// variance modifier `out` and occurs in a contravariant position in a member
/// signature in the body of `D`
/// @author sgrekhov22@gmail.com
// SharedOptions=--enable-experiment=variance,inline-class

class C<out X> {
C({X? x}) {} // No error here
factory C.f({X? x}) => C(x); // And here

void f({X? x}) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

mixin M<out X> {
void f({X? x});
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

enum E<out X> {
e;

const E();
const E.n({X? x}); // No error here

void f({X? x}) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

extension type ET<out X>(X? x) {
ET.n({X? x}) : this.x = x; // No error here

void f({X? x}) {}
// ^
// [analyzer] unspecified
// [cfe] unspecified
}

void main() {
print(C);
print(M);
print(E);
print(ET);
}
89 changes: 89 additions & 0 deletions TypeSystem/type-variance/out_non_covariant_A01_t04.dart
Original file line number Diff line number Diff line change
@@ -0,0 +1,89 @@
// Copyright (c) 2024, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.

/// @assertion Let `D` be the declaration of a class or mixin, enum or a type
/// extension and let `X` be a type parameter declared by `D`.
///
/// If `X` has the variance modifier `out` then it is a compile-time error for
/// `X` to occur in a non-covariant position in a member signature in the body
/// of `D`, except that it is not an error if it occurs in a covariant position
/// in the type annotation of a formal parameter which is covariant (this is a
/// contravariant position in the member signature as a whole).
///
/// In particular, `X` can not be the type of a method parameter (unless it is
/// covariant). It can never be the bound of a type parameter of a generic
/// method
///
/// @description Check that it is a compile-time error if a type parameter has a
/// variance modifier `out` and occurs in a contravariant position in a member
/// signature in the body of `D` (it occurs in a contravariant position in a
/// covariant parameter).
/// @author sgrekhov22@gmail.com
// SharedOptions=--enable-experiment=variance,inline-class

typedef C<T> = T;
typedef ContravariantT<T> = void Function(T);

class A<out X> {
final C<ContravariantT<X>>? v = null;
// ^
// [analyzer] unspecified
// [cfe] unspecified

C<ContravariantT<X>>? f() {}
//^^^^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified

C<ContravariantT<X>>? get g => null;
//^^^^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified
}

mixin M<out X> {
C<ContravariantT<X>>? f() {}
//^^^^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified

C<ContravariantT<X>>? get g => null;
//^^^^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified
}

enum E<out X> {
e;

ContravariantT<X>? f() {}
//^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified

ContravariantT<X>? get g => null;
//^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified
}

extension type ET<out X>(X x) {
ContravariantT<X>? f() {}
//^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified

ContravariantT<X>? get g => null;
//^^^^^^^^^^^^^^^^
// [analyzer] unspecified
// [cfe] unspecified
}

void main() {
print(C);
print(M);
print(E);
print(ET);
}
Loading

0 comments on commit 6dbe0dc

Please sign in to comment.