// Copyright 2013-2014 The Rust Project Developers. See the COPYRIGHT // file at the top-level directory of this distribution and at // http://rust-lang.org/COPYRIGHT. // // Licensed under the Apache License, Version 2.0 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. // ignore-tidy-linelength // ignore-android: FIXME(#10381) // ignore-lldb // compile-flags:-g // gdb-command:set print pretty off // gdb-command:rbreak zzz // gdb-command:run // gdb-command:finish // gdb-command:print stack_unique.value // gdb-check:$1 = 0 // gdb-command:print stack_unique.next.RUST$ENCODED$ENUM$0$Empty.val->value // gdb-check:$2 = 1 // gdb-command:print unique_unique->value // gdb-check:$3 = 2 // gdb-command:print unique_unique->next.RUST$ENCODED$ENUM$0$Empty.val->value // gdb-check:$4 = 3 // gdb-command:print vec_unique[0].value // gdb-check:$5 = 6.5 // gdb-command:print vec_unique[0].next.RUST$ENCODED$ENUM$0$Empty.val->value // gdb-check:$6 = 7.5 // gdb-command:print borrowed_unique->value // gdb-check:$7 = 8.5 // gdb-command:print borrowed_unique->next.RUST$ENCODED$ENUM$0$Empty.val->value // gdb-check:$8 = 9.5 // LONG CYCLE // gdb-command:print long_cycle1.value // gdb-check:$9 = 20 // gdb-command:print long_cycle1.next->value // gdb-check:$10 = 21 // gdb-command:print long_cycle1.next->next->value // gdb-check:$11 = 22 // gdb-command:print long_cycle1.next->next->next->value // gdb-check:$12 = 23 // gdb-command:print long_cycle2.value // gdb-check:$13 = 24 // gdb-command:print long_cycle2.next->value // gdb-check:$14 = 25 // gdb-command:print long_cycle2.next->next->value // gdb-check:$15 = 26 // gdb-command:print long_cycle3.value // gdb-check:$16 = 27 // gdb-command:print long_cycle3.next->value // gdb-check:$17 = 28 // gdb-command:print long_cycle4.value // gdb-check:$18 = 29.5 // gdb-command:print (*****long_cycle_w_anonymous_types).value // gdb-check:$19 = 30 // gdb-command:print (*****((*****long_cycle_w_anonymous_types).next.RUST$ENCODED$ENUM$0$Empty.val)).value // gdb-check:$20 = 31 // gdb-command:continue #![allow(unused_variable)] #![feature(struct_variant)] enum Opt { Empty, Val { val: T } } struct UniqueNode { next: Opt>>, value: T } struct LongCycle1 { next: Box>, value: T, } struct LongCycle2 { next: Box>, value: T, } struct LongCycle3 { next: Box>, value: T, } struct LongCycle4 { next: Option>>, value: T, } struct LongCycleWithAnonymousTypes { next: Opt>>>>>, value: uint, } // This test case makes sure that recursive structs are properly described. The Node structs are // generic so that we can have a new type (that newly needs to be described) for the different // cases. The potential problem with recursive types is that the DI generation algorithm gets // trapped in an endless loop. To make sure, we actually test this in the different cases, we have // to operate on a new type each time, otherwise we would just hit the DI cache for all but the // first case. // The different cases below (stack_*, unique_*, box_*, etc) are set up so that the type description // algorithm will enter the type reference cycle that is created by a recursive definition from a // different context each time. // The "long cycle" cases are constructed to span a longer, indirect recursion cycle between types. // The different locals will cause the DI algorithm to enter the type reference cycle at different // points. fn main() { let stack_unique: UniqueNode = UniqueNode { next: Val { val: box UniqueNode { next: Empty, value: 1_u16, } }, value: 0_u16, }; let unique_unique: Box> = box UniqueNode { next: Val { val: box UniqueNode { next: Empty, value: 3, } }, value: 2, }; let vec_unique: [UniqueNode, ..1] = [UniqueNode { next: Val { val: box UniqueNode { next: Empty, value: 7.5, } }, value: 6.5, }]; let borrowed_unique: &UniqueNode = &UniqueNode { next: Val { val: box UniqueNode { next: Empty, value: 9.5, } }, value: 8.5, }; // LONG CYCLE let long_cycle1: LongCycle1 = LongCycle1 { next: box LongCycle2 { next: box LongCycle3 { next: box LongCycle4 { next: None, value: 23, }, value: 22, }, value: 21 }, value: 20 }; let long_cycle2: LongCycle2 = LongCycle2 { next: box LongCycle3 { next: box LongCycle4 { next: None, value: 26, }, value: 25, }, value: 24 }; let long_cycle3: LongCycle3 = LongCycle3 { next: box LongCycle4 { next: None, value: 28, }, value: 27, }; let long_cycle4: LongCycle4 = LongCycle4 { next: None, value: 29.5, }; // It's important that LongCycleWithAnonymousTypes is encountered only at the end of the // `box` chain. let long_cycle_w_anonymous_types = box box box box box LongCycleWithAnonymousTypes { next: Val { val: box box box box box LongCycleWithAnonymousTypes { next: Empty, value: 31, } }, value: 30 }; zzz(); } fn zzz() {()}