mirror of
https://github.com/dart-lang/sdk
synced 2024-09-16 05:07:52 +00:00
More thorough tests for field promotion in mixins.
These tests address the code review comments on https://dart-review.googlesource.com/c/sdk/+/331750. Change-Id: I79cd14d9cc3228011ba81638b4c1bafdcb8351d4 Reviewed-on: https://dart-review.googlesource.com/c/sdk/+/332746 Reviewed-by: Lasse Nielsen <lrn@google.com> Commit-Queue: Paul Berry <paulberry@google.com> Reviewed-by: Erik Ernst <eernst@google.com>
This commit is contained in:
parent
62241f9c1e
commit
ff437c9623
|
@ -0,0 +1,74 @@
|
||||||
|
// Copyright (c) 2023, 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.
|
||||||
|
|
||||||
|
// Tests that field promotion logic properly handles promotable abstract fields
|
||||||
|
// declared in mixins.
|
||||||
|
|
||||||
|
// In this test, there are concrete implementations of the field in
|
||||||
|
// question. For another test in which there is *no* concrete implementation of
|
||||||
|
// the field, see `abstract_field_in_mixin_test.dart`.
|
||||||
|
|
||||||
|
// This test exercises both syntactic forms of creating mixin applications
|
||||||
|
// (`class C = B with M;` and `class C extends B with M {}`), since these are
|
||||||
|
// represented differently in the analyzer.
|
||||||
|
|
||||||
|
// This test exercises both the scenario in which the mixin declaration precedes
|
||||||
|
// the application, and the scenario in which it follows it. This ensures that
|
||||||
|
// the order in which the mixin declaration and application are analyzed does
|
||||||
|
// not influence the behavior.
|
||||||
|
|
||||||
|
// SharedOptions=--enable-experiment=inference-update-2
|
||||||
|
|
||||||
|
import '../static_type_helper.dart';
|
||||||
|
|
||||||
|
abstract class C1 = Object with M;
|
||||||
|
|
||||||
|
abstract class C2 extends Object with M {}
|
||||||
|
|
||||||
|
mixin M {
|
||||||
|
abstract final int? _field;
|
||||||
|
}
|
||||||
|
|
||||||
|
abstract class C3 = Object with M;
|
||||||
|
|
||||||
|
abstract class C4 extends Object with M {}
|
||||||
|
|
||||||
|
void test(C1 c1, C2 c2, C3 c3, C4 c4) {
|
||||||
|
if (c1._field != null) {
|
||||||
|
c1._field.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c2._field != null) {
|
||||||
|
c2._field.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c3._field != null) {
|
||||||
|
c3._field.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c4._field != null) {
|
||||||
|
c4._field.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ConcreteC1 extends C1 {
|
||||||
|
final int? _field;
|
||||||
|
ConcreteC1(this._field);
|
||||||
|
}
|
||||||
|
|
||||||
|
class ConcreteC2 extends C2 {
|
||||||
|
final int? _field;
|
||||||
|
ConcreteC2(this._field);
|
||||||
|
}
|
||||||
|
|
||||||
|
class ConcreteC3 extends C3 {
|
||||||
|
final int? _field;
|
||||||
|
ConcreteC3(this._field);
|
||||||
|
}
|
||||||
|
|
||||||
|
class ConcreteC4 extends C4 {
|
||||||
|
final int? _field;
|
||||||
|
ConcreteC4(this._field);
|
||||||
|
}
|
||||||
|
|
||||||
|
main() {
|
||||||
|
test(ConcreteC1(0), ConcreteC2(0), ConcreteC3(0), ConcreteC4(0));
|
||||||
|
}
|
|
@ -5,6 +5,14 @@
|
||||||
// Tests that field promotion logic properly handles promotable abstract fields
|
// Tests that field promotion logic properly handles promotable abstract fields
|
||||||
// declared in mixins.
|
// declared in mixins.
|
||||||
|
|
||||||
|
// In this test, there is no concrete implementation of the field in
|
||||||
|
// question. As such, it doesn't really reflect the way an abstract private
|
||||||
|
// field would be used in real-world code, but it's useful for making sure that
|
||||||
|
// an unimplemented abstract private field doesn't cause the analyzer or front
|
||||||
|
// end to misbehave. For another test in which there *is* a concrete
|
||||||
|
// implementation of the field, see
|
||||||
|
// `abstract_field_in_mixin_implemented_test.dart`.
|
||||||
|
|
||||||
// This test exercises both syntactic forms of creating mixin applications
|
// This test exercises both syntactic forms of creating mixin applications
|
||||||
// (`class C = B with M;` and `class C extends B with M {}`), since these are
|
// (`class C = B with M;` and `class C extends B with M {}`), since these are
|
||||||
// represented differently in the analyzer.
|
// represented differently in the analyzer.
|
||||||
|
|
|
@ -0,0 +1,156 @@
|
||||||
|
// Copyright (c) 2023, 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.
|
||||||
|
|
||||||
|
// Tests promotability for final private fields declared in mixins.
|
||||||
|
//
|
||||||
|
// A private final field declared in a mixin is promotable unless:
|
||||||
|
//
|
||||||
|
// - There is a non-final field with the same name declared elsewhere in the
|
||||||
|
// library.
|
||||||
|
//
|
||||||
|
// - There is a concrete getter with the same name declared elsewhere in the
|
||||||
|
// library.
|
||||||
|
//
|
||||||
|
// - There is a concrete class elsewhere in the library that implicitly contains
|
||||||
|
// a noSuchMethod-forwarding getter with the same name.
|
||||||
|
//
|
||||||
|
// This test exercises both ordinary final fields and late final fields.
|
||||||
|
|
||||||
|
// SharedOptions=--enable-experiment=inference-update-2
|
||||||
|
|
||||||
|
import '../static_type_helper.dart';
|
||||||
|
|
||||||
|
// Main test mixin.
|
||||||
|
mixin class M {
|
||||||
|
// Promotable, no conflicting declarations.
|
||||||
|
final int? _nonLate1 = 0;
|
||||||
|
late final int? _late1 = 0;
|
||||||
|
// Not promotable due to same-named non-final field.
|
||||||
|
final int? _nonLate2 = 0;
|
||||||
|
late final int? _late2 = 0;
|
||||||
|
// Not promotable due to same-named getter declaration.
|
||||||
|
final int? _nonLate3 = 0;
|
||||||
|
late final int? _late3 = 0;
|
||||||
|
// Not promotable due to same-named nSM-forwarder.
|
||||||
|
final int? _nonLate4 = 0;
|
||||||
|
late final int? _late4 = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Classes mixing in the main test mixin.
|
||||||
|
class C1 extends Object with M {}
|
||||||
|
|
||||||
|
class C2 = Object with M;
|
||||||
|
|
||||||
|
// Interfering declarations.
|
||||||
|
class C3 implements C4 {
|
||||||
|
// Any non-final field inhibits promotion, since it's not stable.
|
||||||
|
int? _nonLate2;
|
||||||
|
int? _late2;
|
||||||
|
// Any concrete getter inhibits promotion, since it's assumed to not be
|
||||||
|
// stable.
|
||||||
|
int? get _nonLate3 => 0;
|
||||||
|
int? get _late3 => 0;
|
||||||
|
// Any noSuchMethod-forwarding getter inhibits promotion, since the
|
||||||
|
// implementation of noSuchMethod is assumeb to not be stable. (Requires that
|
||||||
|
// the class be concrete and fail to implement a part of its interface; such a
|
||||||
|
// class is only allowed if it contains or inherits a noSuchMethod
|
||||||
|
// declaration).
|
||||||
|
noSuchMethod(Invocation invocation) => 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
class C4 {
|
||||||
|
final int? _nonLate4 = 0;
|
||||||
|
final int? _late4 = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void testPromotionOK(M m, C1 c1, C2 c2) {
|
||||||
|
if (m._nonLate1 != null) {
|
||||||
|
m._nonLate1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (m._late1 != null) {
|
||||||
|
m._late1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c1._nonLate1 != null) {
|
||||||
|
c1._nonLate1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c1._late1 != null) {
|
||||||
|
c1._late1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c2._nonLate1 != null) {
|
||||||
|
c2._nonLate1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
if (c2._late1 != null) {
|
||||||
|
c2._late1.expectStaticType<Exactly<int>>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void testConflictingNonFinalField(M m, C1 c1, C2 c2) {
|
||||||
|
if (m._nonLate2 != null) {
|
||||||
|
m._nonLate2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (m._late2 != null) {
|
||||||
|
m._late2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._nonLate2 != null) {
|
||||||
|
c1._nonLate2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._late2 != null) {
|
||||||
|
c1._late2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._nonLate2 != null) {
|
||||||
|
c2._nonLate2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._late2 != null) {
|
||||||
|
c2._late2.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void testConflictingGetter(M m, C1 c1, C2 c2) {
|
||||||
|
if (m._nonLate3 != null) {
|
||||||
|
m._nonLate3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (m._late3 != null) {
|
||||||
|
m._late3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._nonLate3 != null) {
|
||||||
|
c1._nonLate3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._late3 != null) {
|
||||||
|
c1._late3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._nonLate3 != null) {
|
||||||
|
c2._nonLate3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._late3 != null) {
|
||||||
|
c2._late3.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void testConflictingNSMForwardingGetter(M m, C1 c1, C2 c2) {
|
||||||
|
if (m._nonLate4 != null) {
|
||||||
|
m._nonLate4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (m._late4 != null) {
|
||||||
|
m._late4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._nonLate4 != null) {
|
||||||
|
c1._nonLate4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c1._late4 != null) {
|
||||||
|
c1._late4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._nonLate4 != null) {
|
||||||
|
c2._nonLate4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
if (c2._late4 != null) {
|
||||||
|
c2._late4.expectStaticType<Exactly<int?>>();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
main() {
|
||||||
|
testPromotionOK(M(), C1(), C2());
|
||||||
|
testConflictingNonFinalField(M(), C1(), C2());
|
||||||
|
testConflictingGetter(M(), C1(), C2());
|
||||||
|
testConflictingNSMForwardingGetter(M(), C1(), C2());
|
||||||
|
}
|
Loading…
Reference in a new issue