153 lines
4.5 KiB
Rust
153 lines
4.5 KiB
Rust
// Test where we change a type definition by adding a field. Fns with
|
|
// this type in their signature are recompiled, as are their callers.
|
|
// Fns with that type used only in their body are also recompiled, but
|
|
// their callers are not.
|
|
|
|
// revisions:cfail1 cfail2
|
|
// compile-flags: -Z query-dep-graph
|
|
// build-pass
|
|
|
|
#![feature(rustc_attrs)]
|
|
#![feature(stmt_expr_attributes)]
|
|
#![allow(dead_code)]
|
|
#![crate_type = "rlib"]
|
|
|
|
// These are expected to require codegen.
|
|
#![rustc_partition_codegened(module="struct_point-point", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-fn_with_type_in_sig", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-call_fn_with_type_in_sig", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-fn_with_type_in_body", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-fn_make_struct", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-fn_read_field", cfg="cfail2")]
|
|
#![rustc_partition_codegened(module="struct_point-fn_write_field", cfg="cfail2")]
|
|
|
|
#![rustc_partition_reused(module="struct_point-call_fn_with_type_in_body", cfg="cfail2")]
|
|
|
|
pub mod point {
|
|
#[cfg(cfail1)]
|
|
pub struct Point {
|
|
pub x: f32,
|
|
pub y: f32,
|
|
}
|
|
|
|
#[cfg(cfail2)]
|
|
pub struct Point {
|
|
pub x: f32,
|
|
pub y: f32,
|
|
pub z: f32,
|
|
}
|
|
|
|
impl Point {
|
|
pub fn origin() -> Point {
|
|
#[cfg(cfail1)]
|
|
return Point { x: 0.0, y: 0.0 };
|
|
|
|
#[cfg(cfail2)]
|
|
return Point { x: 0.0, y: 0.0, z: 0.0 };
|
|
}
|
|
|
|
pub fn total(&self) -> f32 {
|
|
#[cfg(cfail1)]
|
|
return self.x + self.y;
|
|
|
|
#[cfg(cfail2)]
|
|
return self.x + self.y + self.z;
|
|
}
|
|
|
|
pub fn x(&self) -> f32 {
|
|
self.x
|
|
}
|
|
}
|
|
}
|
|
|
|
/// A function that has the changed type in its signature; must currently be
|
|
/// rebuilt.
|
|
///
|
|
/// You could imagine that, in the future, if the change were
|
|
/// sufficiently "private", we might not need to type-check again.
|
|
/// Rebuilding is probably always necessary since the layout may be
|
|
/// affected.
|
|
pub mod fn_with_type_in_sig {
|
|
use point::Point;
|
|
|
|
#[rustc_clean(except="typeck,fn_sig,optimized_mir", cfg="cfail2")]
|
|
pub fn boop(p: Option<&Point>) -> f32 {
|
|
p.map(|p| p.total()).unwrap_or(0.0)
|
|
}
|
|
}
|
|
|
|
/// Call a function that has the changed type in its signature; this
|
|
/// currently must also be rebuilt.
|
|
///
|
|
/// You could imagine that, in the future, if the change were
|
|
/// sufficiently "private", we might not need to type-check again.
|
|
/// Rebuilding is probably always necessary since the layout may be
|
|
/// affected.
|
|
pub mod call_fn_with_type_in_sig {
|
|
use fn_with_type_in_sig;
|
|
|
|
#[rustc_clean(except="typeck,optimized_mir", cfg="cfail2")]
|
|
pub fn bip() -> f32 {
|
|
fn_with_type_in_sig::boop(None)
|
|
}
|
|
}
|
|
|
|
/// A function that uses the changed type, but only in its body, not its
|
|
/// signature.
|
|
///
|
|
/// You could imagine that, in the future, if the change were
|
|
/// sufficiently "private", we might not need to type-check again.
|
|
/// Rebuilding is probably always necessary since the layout may be
|
|
/// affected.
|
|
pub mod fn_with_type_in_body {
|
|
use point::Point;
|
|
|
|
#[rustc_clean(except="typeck,optimized_mir", cfg="cfail2")]
|
|
pub fn boop() -> f32 {
|
|
Point::origin().total()
|
|
}
|
|
}
|
|
|
|
/// A function `X` that calls a function `Y`, where `Y` uses the changed type in its
|
|
/// body. In this case, the effects of the change should be contained
|
|
/// to `Y`; `X` should not have to be rebuilt, nor should it need to be
|
|
/// type-checked again.
|
|
pub mod call_fn_with_type_in_body {
|
|
use fn_with_type_in_body;
|
|
|
|
#[rustc_clean(cfg="cfail2")]
|
|
pub fn bip() -> f32 {
|
|
fn_with_type_in_body::boop()
|
|
}
|
|
}
|
|
|
|
/// A function item that makes an instance of `Point` but does not invoke methods.
|
|
pub mod fn_make_struct {
|
|
use point::Point;
|
|
|
|
#[rustc_clean(except="typeck,fn_sig,optimized_mir", cfg="cfail2")]
|
|
pub fn make_origin(p: Point) -> Point {
|
|
Point { ..p }
|
|
}
|
|
}
|
|
|
|
/// A function item that reads fields from `Point` but does not invoke methods.
|
|
pub mod fn_read_field {
|
|
use point::Point;
|
|
|
|
#[rustc_clean(except="typeck,fn_sig,optimized_mir", cfg="cfail2")]
|
|
pub fn get_x(p: Point) -> f32 {
|
|
p.x
|
|
}
|
|
}
|
|
|
|
/// A function item that writes to a field of `Point` but does not invoke methods.
|
|
pub mod fn_write_field {
|
|
use point::Point;
|
|
|
|
#[rustc_clean(except="typeck,fn_sig,optimized_mir", cfg="cfail2")]
|
|
pub fn inc_x(p: &mut Point) {
|
|
p.x += 1.0;
|
|
}
|
|
}
|