Skip to content

Commit e9c6b0a

Browse files
committed
Fix panic=abort when compiling with plugins
Closes #2738
1 parent f4a23e7 commit e9c6b0a

File tree

3 files changed

+141
-1
lines changed

3 files changed

+141
-1
lines changed

src/cargo/ops/cargo_rustc/context.rs

+37
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@ pub struct Context<'a, 'cfg: 'a> {
3939
pub build_config: BuildConfig,
4040
pub build_scripts: HashMap<Unit<'a>, Arc<BuildScripts>>,
4141
pub links: Links<'a>,
42+
pub used_in_plugin: HashSet<Unit<'a>>,
4243

4344
host: Layout,
4445
target: Option<Layout>,
@@ -91,6 +92,7 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
9192
build_scripts: HashMap::new(),
9293
build_explicit_deps: HashMap::new(),
9394
links: Links::new(),
95+
used_in_plugin: HashSet::new(),
9496
})
9597
}
9698

@@ -235,6 +237,41 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
235237
Ok(())
236238
}
237239

240+
/// Builds up the `used_in_plugin` internal to this context from the list of
241+
/// top-level units.
242+
///
243+
/// This will recursively walk `units` and all of their dependencies to
244+
/// determine which crate are going to be used in plugins or not.
245+
pub fn build_used_in_plugin_map(&mut self, units: &[Unit<'a>])
246+
-> CargoResult<()> {
247+
let mut visited = HashSet::new();
248+
for unit in units {
249+
try!(self.walk_used_in_plugin_map(unit,
250+
unit.target.for_host(),
251+
&mut visited));
252+
}
253+
Ok(())
254+
}
255+
256+
fn walk_used_in_plugin_map(&mut self,
257+
unit: &Unit<'a>,
258+
is_plugin: bool,
259+
visited: &mut HashSet<(Unit<'a>, bool)>)
260+
-> CargoResult<()> {
261+
if !visited.insert((*unit, is_plugin)) {
262+
return Ok(())
263+
}
264+
if is_plugin {
265+
self.used_in_plugin.insert(*unit);
266+
}
267+
for unit in try!(self.dep_targets(unit)) {
268+
try!(self.walk_used_in_plugin_map(&unit,
269+
is_plugin || unit.target.for_host(),
270+
visited));
271+
}
272+
Ok(())
273+
}
274+
238275
/// Returns the appropriate directory layout for either a plugin or not.
239276
pub fn layout(&self, unit: &Unit) -> LayoutProxy {
240277
let primary = unit.pkg.package_id() == self.resolve.root();

src/cargo/ops/cargo_rustc/mod.rs

+14-1
Original file line numberDiff line numberDiff line change
@@ -89,6 +89,7 @@ pub fn compile_targets<'a, 'cfg: 'a>(ws: &Workspace<'cfg>,
8989

9090
try!(cx.prepare());
9191
try!(cx.probe_target_info(&units));
92+
try!(cx.build_used_in_plugin_map(&units));
9293
try!(custom_build::build_map(&mut cx, &units));
9394

9495
for unit in units.iter() {
@@ -512,8 +513,20 @@ fn build_base_args(cx: &Context,
512513
cmd.arg("-C").arg(&format!("opt-level={}", opt_level));
513514
}
514515

516+
// If a panic mode was configured *and* we're not ever going to be used in a
517+
// plugin, then we can compile with that panic mode.
518+
//
519+
// If we're used in a plugin then we'll eventually be linked to libsyntax
520+
// most likely which isn't compiled with a custom panic mode, so we'll just
521+
// get an error if we actually compile with that. This fixes `panic=abort`
522+
// crates which have plugin dependencies, but unfortunately means that
523+
// dependencies shared between the main application and plugins must be
524+
// compiled without `panic=abort`. This isn't so bad, though, as the main
525+
// application will still be compiled with `panic=abort`.
515526
if let Some(panic) = panic.as_ref() {
516-
cmd.arg("-C").arg(format!("panic={}", panic));
527+
if !cx.used_in_plugin.contains(unit) {
528+
cmd.arg("-C").arg(format!("panic={}", panic));
529+
}
517530
}
518531

519532
// Disable LTO for host builds as prefer_dynamic and it are mutually

tests/plugins.rs

+90
Original file line numberDiff line numberDiff line change
@@ -274,3 +274,93 @@ fn native_plugin_dependency_with_custom_ar_linker() {
274274
[ERROR] could not exec the linker [..]
275275
"));
276276
}
277+
278+
#[test]
279+
fn panic_abort_plugins() {
280+
if !is_nightly() {
281+
return
282+
}
283+
284+
let bar = project("bar")
285+
.file("Cargo.toml", r#"
286+
[package]
287+
name = "bar"
288+
version = "0.0.1"
289+
authors = []
290+
291+
[profile.dev]
292+
panic = 'abort'
293+
294+
[dependencies]
295+
foo = { path = "foo" }
296+
"#)
297+
.file("src/lib.rs", "")
298+
.file("foo/Cargo.toml", r#"
299+
[package]
300+
name = "foo"
301+
version = "0.0.1"
302+
authors = []
303+
304+
[lib]
305+
plugin = true
306+
"#)
307+
.file("foo/src/lib.rs", r#"
308+
#![feature(rustc_private)]
309+
extern crate syntax;
310+
"#);
311+
312+
assert_that(bar.cargo_process("build"),
313+
execs().with_status(0));
314+
}
315+
316+
#[test]
317+
fn shared_panic_abort_plugins() {
318+
if !is_nightly() {
319+
return
320+
}
321+
322+
let bar = project("top")
323+
.file("Cargo.toml", r#"
324+
[package]
325+
name = "top"
326+
version = "0.0.1"
327+
authors = []
328+
329+
[profile.dev]
330+
panic = 'abort'
331+
332+
[dependencies]
333+
foo = { path = "foo" }
334+
bar = { path = "bar" }
335+
"#)
336+
.file("src/lib.rs", "
337+
extern crate bar;
338+
")
339+
.file("foo/Cargo.toml", r#"
340+
[package]
341+
name = "foo"
342+
version = "0.0.1"
343+
authors = []
344+
345+
[lib]
346+
plugin = true
347+
348+
[dependencies]
349+
bar = { path = "../bar" }
350+
"#)
351+
.file("foo/src/lib.rs", r#"
352+
#![feature(rustc_private)]
353+
extern crate syntax;
354+
extern crate bar;
355+
"#)
356+
.file("bar/Cargo.toml", r#"
357+
[package]
358+
name = "bar"
359+
version = "0.0.1"
360+
authors = []
361+
"#)
362+
.file("bar/src/lib.rs", "");
363+
364+
assert_that(bar.cargo_process("build"),
365+
execs().with_status(0));
366+
}

0 commit comments

Comments
 (0)