| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- // Part of the Carbon Language project, under the Apache License v2.0 with LLVM
- // Exceptions. See /LICENSE for license information.
- // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
- //
- // INCLUDE-FILE: toolchain/testing/testdata/min_prelude/convert.carbon
- //
- // AUTOUPDATE
- // TIP: To test this file alone, run:
- // TIP: bazel test //toolchain/testing:file_test --test_arg=--file_tests=toolchain/check/testdata/impl/lookup/specialization_poison.carbon
- // TIP: To dump output, run:
- // TIP: bazel run //toolchain/testing:file_test -- --dump_output --file_tests=toolchain/check/testdata/impl/lookup/specialization_poison.carbon
- // --- fail_final_poisoned_concrete_query_in_specific.carbon
- library "[[@TEST_NAME]]";
- interface I {
- let T:! type;
- fn F[self: Self]() -> T;
- }
- impl forall [U:! type] U as I where .T = () {
- fn F[self: Self]() -> () { return (); }
- }
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_specific.carbon:[[@LINE+3]]:25: error: found `impl` that would change the result of an earlier use of `C* as I` [PoisonedImplLookupConcreteResult]
- // CHECK:STDERR: fn H[W:! type](v: W) -> W.(I.T) {
- // CHECK:STDERR: ^~~~~~~
- fn H[W:! type](v: W) -> W.(I.T) {
- return v.(I.F)();
- }
- class C { adapt (); }
- fn G(p: C*) -> () {
- // This concrete impl lookup query poisons any further specializations.
- return H(p);
- }
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_specific.carbon:[[@LINE+7]]:1: note: the use would select the `impl` here but it was not found yet [PoisonedImplLookupConcreteResultNoteBadImpl]
- // CHECK:STDERR: impl C* as I where .T = C {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_specific.carbon:[[@LINE-21]]:1: note: the use had selected the `impl` here [PoisonedImplLookupConcreteResultNotePreviousImpl]
- // CHECK:STDERR: impl forall [U:! type] U as I where .T = () {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR:
- impl C* as I where .T = C {
- fn F[self: Self]() -> C { return *self; }
- }
- // --- fail_final_poisoned_concrete_query_in_generic.carbon
- library "[[@TEST_NAME]]";
- interface I {
- let T:! type;
- }
- impl forall [U:! type] U as I where .T = () {}
- class C {}
- fn H[W:! type](v: W) {
- // This concrete impl lookup query poisons any further specializations.
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_generic.carbon:[[@LINE+3]]:10: error: found `impl` that would change the result of an earlier use of `C as I` [PoisonedImplLookupConcreteResult]
- // CHECK:STDERR: let a: C.(I.T) = ();
- // CHECK:STDERR: ^~~~~~~
- let a: C.(I.T) = ();
- }
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_generic.carbon:[[@LINE+7]]:1: note: the use would select the `impl` here but it was not found yet [PoisonedImplLookupConcreteResultNoteBadImpl]
- // CHECK:STDERR: impl C as I where .T = C {}
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR: fail_final_poisoned_concrete_query_in_generic.carbon:[[@LINE-15]]:1: note: the use had selected the `impl` here [PoisonedImplLookupConcreteResultNotePreviousImpl]
- // CHECK:STDERR: impl forall [U:! type] U as I where .T = () {}
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR:
- impl C as I where .T = C {}
- // --- fail_final_poisoned_concrete_query_nested_type_in_self.carbon
- library "[[@TEST_NAME]]";
- interface I {
- let T:! type;
- fn F[self: Self]() -> T;
- }
- class C(U:! type) {}
- impl forall [U:! type] C(U) as I where .T = () {
- fn F[self: Self]() -> () { return (); }
- }
- fn G(c: C(())) -> () {
- // This concrete impl lookup query poisons any further specializations.
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_self.carbon:[[@LINE+3]]:10: error: found `impl` that would change the result of an earlier use of `C(()) as I` [PoisonedImplLookupConcreteResult]
- // CHECK:STDERR: return c.(I.F)();
- // CHECK:STDERR: ^~~~~~~
- return c.(I.F)();
- }
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_self.carbon:[[@LINE+7]]:1: note: the use would select the `impl` here but it was not found yet [PoisonedImplLookupConcreteResultNoteBadImpl]
- // CHECK:STDERR: impl C(()) as I where .T = {} {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_self.carbon:[[@LINE-15]]:1: note: the use had selected the `impl` here [PoisonedImplLookupConcreteResultNotePreviousImpl]
- // CHECK:STDERR: impl forall [U:! type] C(U) as I where .T = () {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR:
- impl C(()) as I where .T = {} {
- fn F[self: Self]() -> {} { return {}; }
- }
- // --- fail_final_poisoned_concrete_query_nested_type_in_interface.carbon
- library "[[@TEST_NAME]]";
- interface I(U:! type) {
- let T:! type;
- fn F[self: Self]() -> T;
- }
- impl forall [U:! type] U as I(U) where .T = () {
- fn F[self: Self]() -> () { return (); }
- }
- class C {}
- fn G(c: C) -> () {
- // This concrete impl lookup query poisons any further specializations.
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_interface.carbon:[[@LINE+3]]:10: error: found `impl` that would change the result of an earlier use of `C as I(C)` [PoisonedImplLookupConcreteResult]
- // CHECK:STDERR: return c.(I(C).F)();
- // CHECK:STDERR: ^~~~~~~~~~
- return c.(I(C).F)();
- }
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_interface.carbon:[[@LINE+7]]:1: note: the use would select the `impl` here but it was not found yet [PoisonedImplLookupConcreteResultNoteBadImpl]
- // CHECK:STDERR: impl C as I(C) where .T = {} {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR: fail_final_poisoned_concrete_query_nested_type_in_interface.carbon:[[@LINE-17]]:1: note: the use had selected the `impl` here [PoisonedImplLookupConcreteResultNotePreviousImpl]
- // CHECK:STDERR: impl forall [U:! type] U as I(U) where .T = () {
- // CHECK:STDERR: ^~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- // CHECK:STDERR:
- impl C as I(C) where .T = {} {
- fn F[self: Self]() -> {} { return {}; }
- }
- // --- todo_fail_final_poisoned_by_generic_query.carbon
- library "[[@TEST_NAME]]";
- interface I {
- let T:! type;
- fn F[self: Self]() -> T;
- }
- impl forall [U:! type] U as I where .T = () {
- fn F[self: Self]() -> () { return (); }
- }
- fn H[W:! type](v: W) -> W.(I.T) {
- return v.(I.F)();
- }
- // This function could return a concrete `X` if it saw the `final` impl below.
- fn G[X:! type](p: X*) -> (X*).(I.T) {
- return H(p);
- }
- // TODO: This should be diagnosed as poisoned, as H() uses an associated
- // constant from the `impl` which was treated as a symbolic, but this would
- // change it to be a concrete type.
- final impl forall [V:! Core.Copy] V* as I where .T = V {
- fn F[self: Self]() -> V { return *self; }
- }
|