// Test that we properly detect the cycle amongst the traits // here and report an error. use std::panic::RefUnwindSafe; trait Database { type Storage; } trait HasQueryGroup {} trait Query { type Data; } trait SourceDatabase { fn parse(&self) { loop {} } } struct ParseQuery; struct RootDatabase { _runtime: Runtime, } struct Runtime { _storage: Box, } struct SalsaStorage { _parse: >::Data, //~ ERROR overflow } impl Database for RootDatabase { //~ ERROR overflow type Storage = SalsaStorage; } impl HasQueryGroup for RootDatabase {} impl Query for ParseQuery where DB: SourceDatabase, DB: Database, { type Data = RootDatabase; } impl SourceDatabase for T where T: RefUnwindSafe, T: HasQueryGroup, { } pub(crate) fn goto_implementation(db: &RootDatabase) -> u32 { // This is not satisfied: // // - `RootDatabase: SourceDatabase` // - requires `RootDatabase: RefUnwindSafe` + `RootDatabase: HasQueryGroup` // - `RootDatabase: RefUnwindSafe` // - requires `Runtime: RefUnwindSafe` // - `Runtime: RefUnwindSafe` // - requires `DB::Storage: RefUnwindSafe` (`SalsaStorage: RefUnwindSafe`) // - `SalsaStorage: RefUnwindSafe` // - requires `>::Data: RefUnwindSafe`, // which means `ParseQuery: Query` // - `ParseQuery: Query` // - requires `RootDatabase: SourceDatabase`, // - `RootDatabase: SourceDatabase` is already on the stack, so we have a // cycle with non-coinductive participants // // we used to fail to report an error here because we got the // caching wrong. SourceDatabase::parse(db); 22 } fn main() {}