blob: 4feba4698ba2054b841410daab5581bd7370b910 [file] [log] [blame]
// Copyright (c) 2018, 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.
import "package:expect/expect.dart";
class A {}
class B extends A {}
class B2 extends A {}
class C {
void f(B x) {}
}
abstract class I {
void f(covariant A x);
}
// This class contains a forwarding stub for f to allow it to satisfy the
// interface I, while still ensuring that the x argument is type checked before
// C.f is executed.
//
// For purposes of static type checking, the interface of the class D is
// considered to contain a method f with signature (A) -> void. For purposes of
// runtime behavior, a tearoff of D.f is considered to have the reified runtime
// type (Object) -> void.
class D extends C implements I {}
main() {
var d = new D();
B2 b2Null = null;
B2 b2 = new B2();
// Since the compile-time type of D.f is (A) -> void, it is assignable to (B2)
// -> void. Since the runtime type is (Object) -> void, the assignment is
// allowed at runtime as well.
void Function(B2) g = d.f;
// However, the tear-off performs a runtime check of its argument, so it
// accepts a value of `null`, but it does not accept a value whose runtime
// type is B2.
g(b2Null);
Expect.throwsTypeError(() {
g(b2);
});
}