diff options
Diffstat (limited to 'tests/incremental/change_add_field')
-rw-r--r-- | tests/incremental/change_add_field/struct_point.rs | 152 |
1 files changed, 152 insertions, 0 deletions
diff --git a/tests/incremental/change_add_field/struct_point.rs b/tests/incremental/change_add_field/struct_point.rs new file mode 100644 index 000000000..3308ea562 --- /dev/null +++ b/tests/incremental/change_add_field/struct_point.rs @@ -0,0 +1,152 @@ +// 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; + } +} |