Recognize infinite recursion caused by {self}

This commit is contained in:
David Tolnay 2024-11-04 22:18:46 -05:00
parent e1fa5190b4
commit 6a0eb08569
No known key found for this signature in database
GPG key ID: F9BA143B95FF6D82
4 changed files with 68 additions and 18 deletions

View file

@ -102,29 +102,40 @@ impl Display<'_> {
MemberUnraw::Unnamed(index) => format_ident!("_{}", index),
MemberUnraw::Named(ident) => ident.to_local(),
});
if let Some(&field) = member_index.get(&member) {
let end_spec = match read.find('}') {
Some(end_spec) => end_spec,
None => return Ok(()),
};
let bound = match read[..end_spec].chars().next_back() {
Some('?') => Trait::Debug,
Some('o') => Trait::Octal,
Some('x') => Trait::LowerHex,
Some('X') => Trait::UpperHex,
Some('p') => Trait::Pointer,
Some('b') => Trait::Binary,
Some('e') => Trait::LowerExp,
Some('E') => Trait::UpperExp,
Some(_) => Trait::Display,
None => {
let end_spec = match read.find('}') {
Some(end_spec) => end_spec,
None => return Ok(()),
};
let bound = match read[..end_spec].chars().next_back() {
Some('?') => Trait::Debug,
Some('o') => Trait::Octal,
Some('x') => Trait::LowerHex,
Some('X') => Trait::UpperHex,
Some('p') => Trait::Pointer,
Some('b') => Trait::Binary,
Some('e') => Trait::LowerExp,
Some('E') => Trait::UpperExp,
Some(_) => Trait::Display,
None => {
if member_index.contains_key(&member) {
has_bonus_display = true;
binding_value.extend(quote_spanned!(span=> .as_display()));
Trait::Display
}
};
Trait::Display
}
};
if let Some(&field) = member_index.get(&member) {
implied_bounds.insert((field, bound));
}
if member == *"self" && bound == Trait::Display {
binding_value = quote_spanned!(member.span()=>
{
#[warn(unconditional_recursion)]
fn _fmt() { _fmt() }
#member
}
);
}
bindings.push((local, binding_value));
}

View file

@ -101,6 +101,15 @@ impl PartialEq for MemberUnraw {
}
}
impl PartialEq<str> for MemberUnraw {
fn eq(&self, other: &str) -> bool {
match self {
MemberUnraw::Named(this) => this == other,
MemberUnraw::Unnamed(_) => false,
}
}
}
impl Hash for MemberUnraw {
fn hash<H: Hasher>(&self, hasher: &mut H) {
match self {

View file

@ -0,0 +1,9 @@
use thiserror::Error;
#[derive(Error, Debug)]
#[error("{self}")]
pub struct Error;
fn main() {
@//fail
}

View file

@ -0,0 +1,21 @@
error: expected expression, found `@`
--> tests/ui/unconditional-recursion.rs:8:5
|
8 | @//fail
| ^ expected expression
warning: function cannot return without recursing
--> tests/ui/unconditional-recursion.rs:4:9
|
4 | #[error("{self}")]
| ^^^^^^^^
| |
| cannot return without recursing
| recursive call site
|
= help: a `loop` may express intention better if this is on purpose
note: the lint level is defined here
--> tests/ui/unconditional-recursion.rs:4:9
|
4 | #[error("{self}")]
| ^^^^^^^^