| // 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 | 
 | //@ ignore-backends: gcc | 
 |  | 
 | #![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; | 
 |     } | 
 | } |