forked from rust-lang/rust
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Rollup merge of rust-lang#61488 - matthewjasper:fix-nll-typeck-ices, …
…r=pnkfelix Fix NLL typeck ICEs * Don't ICE when a type containing a region is constrained by nothing * Don't ICE trying to normalize a type in a `ParamEnv` containing global bounds. To explain what was happening in the `issue-61311-normalize.rs` case: * When borrow checking the `the_fn` in the last `impl` we would try to normalize `Self::Proj` (`<Unit as HasProjFn>::Proj`). * We would find the `impl` that we're checking and and check its `where` clause. * This would need us to check `<Box<dyn Obj + 'static> as HasProj>::Proj: Bound` * We find two possible implementations, the blanket impl and the bound in our `ParamEnv`. * The bound in our `ParamEnv` was canonicalized, so we don't see it as a global bound. As such we prefer it to the `impl`. * This means that we cannot normalize `<Box<dyn Obj + 'static> as HasProj>::Proj` to `Unit`. * The `<Box<dyn Obj + 'static> as HasProj>::Proj: Bound` bound, which looks like it should be in our `ParamEnv` has been normalized to `Unit: Bound`. * We fail to prove `<Box<dyn Obj + 'static> as HasProj>::Proj: Bound`. * We ICE, since we believe typeck have errored. Closes rust-lang#61311 Closes rust-lang#61315 Closes rust-lang#61320 r? @pnkfelix cc @nikomatsakis
- Loading branch information
Showing
6 changed files
with
222 additions
and
4 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,11 @@ | ||
// Regression test for #61315 | ||
// | ||
// `dyn T:` is lowered to `dyn T: ReEmpty` - check that we don't ICE in NLL for | ||
// the unexpected region. | ||
|
||
// compile-pass | ||
|
||
trait T {} | ||
fn f() where dyn T: {} | ||
|
||
fn main() {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,34 @@ | ||
// Regression test for #61311 | ||
// We would ICE after failing to normalize `Self::Proj` in the `impl` below. | ||
|
||
// compile-pass | ||
|
||
pub struct Unit; | ||
trait Obj {} | ||
|
||
trait Bound {} | ||
impl Bound for Unit {} | ||
|
||
pub trait HasProj { | ||
type Proj; | ||
} | ||
|
||
impl<T> HasProj for T { | ||
type Proj = Unit; | ||
} | ||
|
||
trait HasProjFn { | ||
type Proj; | ||
fn the_fn(_: Self::Proj); | ||
} | ||
|
||
impl HasProjFn for Unit | ||
where | ||
Box<dyn Obj + 'static>: HasProj, | ||
<Box<dyn Obj + 'static> as HasProj>::Proj: Bound, | ||
{ | ||
type Proj = Unit; | ||
fn the_fn(_: Self::Proj) {} | ||
} | ||
|
||
fn main() {} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,160 @@ | ||
// Regression test for #61320 | ||
// This is the same issue as #61311, just a larger test case. | ||
|
||
// compile-pass | ||
|
||
pub struct AndThen<A, B, F> | ||
where | ||
A: Future, | ||
B: IntoFuture, | ||
{ | ||
state: (A, B::Future, F), | ||
} | ||
|
||
pub struct FutureResult<T, E> { | ||
inner: Option<Result<T, E>>, | ||
} | ||
|
||
impl<T, E> Future for FutureResult<T, E> { | ||
type Item = T; | ||
type Error = E; | ||
|
||
fn poll(&mut self) -> Poll<T, E> { | ||
unimplemented!() | ||
} | ||
} | ||
|
||
pub type Poll<T, E> = Result<T, E>; | ||
|
||
impl<A, B, F> Future for AndThen<A, B, F> | ||
where | ||
A: Future, | ||
B: IntoFuture<Error = A::Error>, | ||
F: FnOnce(A::Item) -> B, | ||
{ | ||
type Item = B::Item; | ||
type Error = B::Error; | ||
|
||
fn poll(&mut self) -> Poll<B::Item, B::Error> { | ||
unimplemented!() | ||
} | ||
} | ||
|
||
pub trait Future { | ||
type Item; | ||
|
||
type Error; | ||
|
||
fn poll(&mut self) -> Poll<Self::Item, Self::Error>; | ||
|
||
fn and_then<F, B>(self, f: F) -> AndThen<Self, B, F> | ||
where | ||
F: FnOnce(Self::Item) -> B, | ||
B: IntoFuture<Error = Self::Error>, | ||
Self: Sized, | ||
{ | ||
unimplemented!() | ||
} | ||
} | ||
|
||
pub trait IntoFuture { | ||
/// The future that this type can be converted into. | ||
type Future: Future<Item = Self::Item, Error = Self::Error>; | ||
|
||
/// The item that the future may resolve with. | ||
type Item; | ||
/// The error that the future may resolve with. | ||
type Error; | ||
|
||
/// Consumes this object and produces a future. | ||
fn into_future(self) -> Self::Future; | ||
} | ||
|
||
impl<F: Future> IntoFuture for F { | ||
type Future = F; | ||
type Item = F::Item; | ||
type Error = F::Error; | ||
|
||
fn into_future(self) -> F { | ||
self | ||
} | ||
} | ||
|
||
impl<F: ?Sized + Future> Future for ::std::boxed::Box<F> { | ||
type Item = F::Item; | ||
type Error = F::Error; | ||
|
||
fn poll(&mut self) -> Poll<Self::Item, Self::Error> { | ||
(**self).poll() | ||
} | ||
} | ||
|
||
impl<T, E> IntoFuture for Result<T, E> { | ||
type Future = FutureResult<T, E>; | ||
type Item = T; | ||
type Error = E; | ||
|
||
fn into_future(self) -> FutureResult<T, E> { | ||
unimplemented!() | ||
} | ||
} | ||
|
||
struct Request<T>(T); | ||
|
||
trait RequestContext {} | ||
impl<T> RequestContext for T {} | ||
struct NoContext; | ||
impl AsRef<NoContext> for NoContext { | ||
fn as_ref(&self) -> &Self { | ||
&NoContext | ||
} | ||
} | ||
|
||
type BoxedError = Box<dyn std::error::Error + Send + Sync>; | ||
type DefaultFuture<T, E> = Box<dyn Future<Item = T, Error = E> + Send>; | ||
|
||
trait Guard: Sized { | ||
type Result: IntoFuture<Item = Self, Error = BoxedError>; | ||
fn from_request(request: &Request<()>) -> Self::Result; | ||
} | ||
|
||
trait FromRequest: Sized { | ||
type Context; | ||
type Future: Future<Item = Self, Error = BoxedError> + Send; | ||
fn from_request(request: Request<()>) -> Self::Future; | ||
} | ||
|
||
struct MyGuard; | ||
impl Guard for MyGuard { | ||
type Result = Result<Self, BoxedError>; | ||
fn from_request(_request: &Request<()>) -> Self::Result { | ||
Ok(MyGuard) | ||
} | ||
} | ||
|
||
struct Generic<I> { | ||
_inner: I, | ||
} | ||
|
||
impl<I> FromRequest for Generic<I> | ||
where | ||
MyGuard: Guard, | ||
<MyGuard as Guard>::Result: IntoFuture<Item = MyGuard, Error = BoxedError>, | ||
<<MyGuard as Guard>::Result as IntoFuture>::Future: Send, | ||
I: FromRequest<Context = NoContext>, | ||
{ | ||
type Future = DefaultFuture<Self, BoxedError>; | ||
type Context = NoContext; | ||
fn from_request(headers: Request<()>) -> DefaultFuture<Self, BoxedError> { | ||
let _future = <MyGuard as Guard>::from_request(&headers) | ||
.into_future() | ||
.and_then(move |_| { | ||
<I as FromRequest>::from_request(headers) | ||
.into_future() | ||
.and_then(move |fld_inner| Ok(Generic { _inner: fld_inner }).into_future()) | ||
}); | ||
panic!(); | ||
} | ||
} | ||
|
||
fn main() {} |