Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Checking that main's output type implements Termination doesn't normalize associated types #88609

Closed
SkiFire13 opened this issue Sep 3, 2021 · 1 comment · Fixed by #88706
Assignees
Labels
C-bug Category: This is a bug.

Comments

@SkiFire13
Copy link
Contributor

SkiFire13 commented Sep 3, 2021

I tried this code:

#![feature(termination_trait_lib)]

trait Same {
    type Output;
}
impl<T> Same for T {
    type Output = T;
}
type Unit = <() as Same>::Output;

fn test() {
    fn foo() -> Result<Unit, std::io::Error> { todo!() }

    fn assert_impl_termination<T: std::process::Termination>(_: fn() -> T) {}
    // Compiles
    assert_impl_termination(foo);
}

// Doesn't compile
fn main() -> Result<Unit, std::io::Error> {
    todo!()
}

Playground link

I expected to see both test and main compile, instead only test compiles, while trying to compile main gives this error:

error[E0277]: `main` has invalid return type `Result<<() as Same>::Output, std::io::Error>`
  --> src/main.rs:20:14
   |
20 | fn main() -> Result<Unit, std::io::Error> {
   |              ^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `main` can only return types that implement `Termination`
   |
   = help: consider using `()`, or a `Result`
help: consider introducing a `where` bound, but there might be an alternative better way to express this requirement
   |
20 | fn main() -> Result<Unit, std::io::Error> where Result<<() as Same>::Output, std::io::Error>: Termination {
   |                                           +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

For more information about this error, try `rustc --explain E0277`.

Note that the feature attribute is only needed for the bound in assert_impl_termination. The issue with main happens both in stable and nightly, with or without the feature.

Meta

rustc --version --verbose:

rustc 1.54.0 (a178d0322 2021-07-26)
binary: rustc
commit-hash: a178d0322ce20e33eac124758e837cbd80a6f633
commit-date: 2021-07-26
host: x86_64-pc-windows-msvc
release: 1.54.0
LLVM version: 12.0.1
rustc 1.56.0-nightly (29ef6cf16 2021-08-31)
binary: rustc
commit-hash: 29ef6cf1637aa8317f8911f93f14e18d404c1b0e
commit-date: 2021-08-31
host: x86_64-pc-windows-msvc
release: 1.56.0-nightly
LLVM version: 13.0.0
@SkiFire13 SkiFire13 added the C-bug Category: This is a bug. label Sep 3, 2021
@ThePuzzlemaker
Copy link
Contributor

I am going to attempt to work on this. I've already got a solution that seems to work, but it's probably not the best way.

@rustbot claim

Manishearth added a commit to Manishearth/rust that referenced this issue Oct 5, 2021
Normalize associated type projections when checking return type of main

This fixes rust-lang#88609.

Previously, the return type of `fn main()` would not have any associated type projections within normalized before checking if it implements the standard library trait `std::process::Termination`. This commit appears to fix it.

This feels vaguely symptomatic of a problem in the underlying trait solving engine, but I am not sure how I would solve that. I am unsure why the example in rust-lang#88609 with `assert_impl_termination` and `fn foo()` work, but simply `fn main()` doesn't. The way that I solved this is also probably not the best way to do this, so please let me know if there is a better way to do this.

I have added a build-pass regression test for this issue.
@bors bors closed this as completed in 734209e Oct 6, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
C-bug Category: This is a bug.
Projects
None yet
Development

Successfully merging a pull request may close this issue.

2 participants