summaryrefslogtreecommitdiffstats
path: root/tests/incremental/change_add_field/struct_point.rs
diff options
context:
space:
mode:
Diffstat (limited to 'tests/incremental/change_add_field/struct_point.rs')
-rw-r--r--tests/incremental/change_add_field/struct_point.rs152
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;
+ }
+}