Skip to content

Commit

Permalink
Auto merge of #34805 - michaelwoerister:stable-bounds-encoding, r=eddyb
Browse files Browse the repository at this point in the history
tyencode: Make sure that projection bounds are handled in stable order.

Fixes #34796.

r? @alexcrichton
  • Loading branch information
bors committed Jul 14, 2016
2 parents 935bd76 + 5ad5072 commit 3c85f41
Show file tree
Hide file tree
Showing 3 changed files with 74 additions and 1 deletion.
9 changes: 8 additions & 1 deletion src/librustc_metadata/tyencode.rs
Expand Up @@ -407,7 +407,14 @@ pub fn enc_existential_bounds<'a,'tcx>(w: &mut Cursor<Vec<u8>>,

enc_region(w, cx, bs.region_bound);

for tp in &bs.projection_bounds {
// Encode projection_bounds in a stable order
let mut projection_bounds: Vec<_> = bs.projection_bounds
.iter()
.map(|b| (b.item_name().as_str(), b))
.collect();
projection_bounds.sort_by_key(|&(ref name, _)| name.clone());

for tp in projection_bounds.iter().map(|&(_, tp)| tp) {
write!(w, "P");
enc_projection_predicate(w, cx, &tp.0);
}
Expand Down
30 changes: 30 additions & 0 deletions src/test/run-pass/auxiliary/issue34796aux.rs
@@ -0,0 +1,30 @@
// Copyright 2016 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 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

#![crate_type = "lib"]
pub trait Future {
type Item;
type Error;
}

impl Future for u32 {
type Item = ();
type Error = Box<()>;
}

fn foo() -> Box<Future<Item=(), Error=Box<()>>> {
Box::new(0u32)
}

pub fn bar<F, A, B>(_s: F)
where F: Fn(A) -> B,
{
foo();
}
36 changes: 36 additions & 0 deletions src/test/run-pass/issue34796.rs
@@ -0,0 +1,36 @@
// Copyright 2016 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 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// This test case exposes conditions where the encoding of a trait object type
// with projection predicates would differ between this crate and the upstream
// crate, because the predicates were encoded in different order within each
// crate. This led to different symbol hashes of functions using these type,
// which in turn led to linker errors because the two crates would not agree on
// the symbol name.
// The fix was to make the order in which predicates get encoded stable.

// aux-build:issue34796aux.rs
extern crate issue34796aux;

fn mk<T>() -> T { loop {} }

struct Data<T, E> {
data: T,
error: E,
}

fn main() {
issue34796aux::bar(|()| {
Data::<(), std::io::Error> {
data: mk(),
error: mk(),
}
})
}

0 comments on commit 3c85f41

Please sign in to comment.