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

Evaluation overflow with specialization feature #48515

Open
joshlf opened this issue Feb 24, 2018 · 11 comments
Open

Evaluation overflow with specialization feature #48515

joshlf opened this issue Feb 24, 2018 · 11 comments
Labels
A-diagnostics Area: Messages for errors, warnings, and lints A-specialization Area: Trait impl specialization C-enhancement Category: An issue proposing an enhancement or a PR with one. F-specialization `#![feature(specialization)]` T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.

Comments

@joshlf
Copy link
Contributor

joshlf commented Feb 24, 2018

The current nightly compiler gives the error "overflow evaluating requirement" when a type is used as a trait with a default impl. For example:

#![feature(specialization)]

fn main() {
    println!("{}", <(usize) as TypeString>::type_string());
}

trait TypeString {
    fn type_string() -> &'static str;
}

default impl<T> TypeString for T {
    fn type_string() -> &'static str {
        "unknown type"
    }
}

impl TypeString for () {
    fn type_string() -> &'static str {
        "()"
    }
}

(playground link)

...gives the following error:

error[E0275]: overflow evaluating the requirement `usize: TypeString`
 --> src/main.rs:4:20
  |
4 |     println!("{}", <(usize) as TypeString>::type_string());
  |                    ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  |
note: required by `TypeString::type_string`
 --> src/main.rs:8:5
  |
8 |     fn type_string() -> &'static str;
  |     ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^

Note that if we change <(usize) as TypeString>::type_string() to <() as TypeString>::type_string() - () has a specialized impl - it compiles and runs correctly.

@vitalyd
Copy link

vitalyd commented Feb 24, 2018

Note if you move the default keyword to the fn itself, it works. Not sure if this is intentional or not.

@joshlf
Copy link
Contributor Author

joshlf commented Feb 24, 2018

Oh, odd. And it looks like it's allowed to have the default keyword on both the impl and the fn, although that doesn't fix this bug. That's pretty confusing. Does anyone know if they do different things, or are they just synonyms?

@qnighy
Copy link
Contributor

qnighy commented Feb 25, 2018

impl with defaulted items and default impl are different as described in rust-lang/rfcs#1210. However, unfortunately, they had been implemented uniformly before #45404.

@joshlf
Copy link
Contributor Author

joshlf commented Feb 25, 2018

Am I misreading the RFC? It looks like it doesn't define default impl at all:

Why mark default on items rather than the entire impl? Again, this is largely about granularity; it's useful to be able to pin down part of an impl while leaving others open for specialization. Furthermore, while this RFC doesn't propose to do it, we could easily add a shorthand later on in which default impl Trait for Type is sugar for adding default to all items in the impl.

@qnighy
Copy link
Contributor

qnighy commented Feb 26, 2018

Strange. The other parts of the RFC mention default impl with the clear example:

// the `default` qualifier here means (1) not all items are impled
// and (2) those that are can be further specialized
default impl<T: Clone, Rhs> Add<Rhs> for T {
    fn add_assign(&mut self, rhs: R) {
        let tmp = self.clone() + rhs;
        *self = tmp;
    }
}

To me the RFC seems to be self-contradicting a bit.

Anyway, from the example above, we can see that default impls with partial implementation are allowed. The remaining point is whether default impls with full implementation are used to resolve trait obligation. I think it is good to treat them uniformly with partial default impls, as @nikomatsakis says.

@pietroalbini pietroalbini added C-enhancement Category: An issue proposing an enhancement or a PR with one. A-diagnostics Area: Messages for errors, warnings, and lints T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. A-specialization Area: Trait impl specialization labels Feb 27, 2018
@aidanhs
Copy link
Member

aidanhs commented Mar 12, 2018

#![feature(specialization)]
trait Type {
	const FOO: bool;
}
default impl<T> Type for T {
	const FOO: bool = false;
}
fn main() {
    <u8 as Type>::FOO;
}

Another example.

@joshlf
Copy link
Contributor Author

joshlf commented Mar 2, 2019

Any progress on this? I ran into it again. Anybody I should consult on this? I'd like to help debug since this is blocking me on some work.

EDIT: Not blocked anymore; I figured out a workaround.

@athre0z
Copy link
Contributor

athre0z commented Mar 6, 2019

Would you mind sharing that workaround @joshlf?

@joshlf
Copy link
Contributor Author

joshlf commented Mar 6, 2019

It's unfortunately not a general workaround; I just figured out a way to avoid using this feature entirely so it wasn't a problem.

@jonas-schievink jonas-schievink added the F-specialization `#![feature(specialization)]` label Sep 15, 2019
@Avi-D-coder
Copy link
Contributor

Anyone know, where to start fixing this?
The workaround worked for my use case, but it's a little strange and unexpected.
Just adding a message to the error about this issue could help some.

@estk
Copy link
Contributor

estk commented Sep 23, 2022

lol i keep forgetting about this and running into it.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-diagnostics Area: Messages for errors, warnings, and lints A-specialization Area: Trait impl specialization C-enhancement Category: An issue proposing an enhancement or a PR with one. F-specialization `#![feature(specialization)]` T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.
Projects
None yet
Development

No branches or pull requests

9 participants