//! Test for inner statics with the same name. //! //! Before, the path name for all items defined in methods of traits and impls never //! took into account the name of the method. This meant that if you had two statics //! of the same name in two different methods the statics would end up having the //! same symbol named (even after mangling) because the path components leading to //! the symbol were exactly the same (just __extensions__ and the static name). //! //! It turns out that if you add the symbol "A" twice to LLVM, it automatically //! makes the second one "A1" instead of "A". What this meant is that in local crate //! compilations we never found this bug. Even across crates, this was never a //! problem. The problem arises when you have generic methods that don't get //! generated at compile-time of a library. If the statics were re-added to LLVM by //! a client crate of a library in a different order, you would reference different //! constants (the integer suffixes wouldn't be guaranteed to be the same). pub struct A { pub v: T } pub struct B { pub v: T } pub mod test { pub struct A { pub v: T } impl A { pub fn foo(&self) -> isize { static a: isize = 5; return a } pub fn bar(&self) -> isize { static a: isize = 6; return a; } } } impl A { pub fn foo(&self) -> isize { static a: isize = 1; return a } pub fn bar(&self) -> isize { static a: isize = 2; return a; } } impl B { pub fn foo(&self) -> isize { static a: isize = 3; return a } pub fn bar(&self) -> isize { static a: isize = 4; return a; } } pub fn foo() -> isize { let a = A { v: () }; let b = B { v: () }; let c = test::A { v: () }; return a.foo() + a.bar() + b.foo() + b.bar() + c.foo() + c.bar(); }