about summary refs log tree commit diff
path: root/src
diff options
context:
space:
mode:
authorEsteban Küber <esteban@kuber.com.ar>2019-04-17 12:00:35 -0700
committerEsteban Küber <esteban@kuber.com.ar>2019-04-17 12:00:35 -0700
commitd56c82074b4a25df9ae618107af4f11f8dda6238 (patch)
tree8b487df028c28bd5ec3ff2a1312556eb620dcb26 /src
parent70f130954d6fd36fe3e77f4b5b33e5af50489288 (diff)
downloadrust-d56c82074b4a25df9ae618107af4f11f8dda6238.tar.gz
rust-d56c82074b4a25df9ae618107af4f11f8dda6238.zip
Fix ICE on const evaluation of const method
Diffstat (limited to 'src')
-rw-r--r--src/librustc/ty/sty.rs13
-rw-r--r--src/test/ui/issues/issue-54954.rs19
-rw-r--r--src/test/ui/issues/issue-54954.stderr16
3 files changed, 48 insertions, 0 deletions
diff --git a/src/librustc/ty/sty.rs b/src/librustc/ty/sty.rs
index df76e6127e8..40612ab931a 100644
--- a/src/librustc/ty/sty.rs
+++ b/src/librustc/ty/sty.rs
@@ -1007,6 +1007,16 @@ impl<'tcx> FnSig<'tcx> {
     pub fn output(&self) -> Ty<'tcx> {
         self.inputs_and_output[self.inputs_and_output.len() - 1]
     }
+
+    // Create a minimal `FnSig` to be used when encountering a `TyKind::Error` in a fallible method
+    fn fake() -> FnSig<'tcx> {
+        FnSig {
+            inputs_and_output: List::empty(),
+            c_variadic: false,
+            unsafety: hir::Unsafety::Normal,
+            abi: abi::Abi::Rust,
+        }
+    }
 }
 
 pub type PolyFnSig<'tcx> = Binder<FnSig<'tcx>>;
@@ -1955,6 +1965,9 @@ impl<'a, 'gcx, 'tcx> TyS<'tcx> {
                 tcx.fn_sig(def_id).subst(tcx, substs)
             }
             FnPtr(f) => f,
+            Error => {  // ignore errors (#54954)
+                ty::Binder::dummy(FnSig::fake())
+            }
             _ => bug!("Ty::fn_sig() called on non-fn type: {:?}", self)
         }
     }
diff --git a/src/test/ui/issues/issue-54954.rs b/src/test/ui/issues/issue-54954.rs
new file mode 100644
index 00000000000..40045bc758c
--- /dev/null
+++ b/src/test/ui/issues/issue-54954.rs
@@ -0,0 +1,19 @@
+#![feature(const_fn)]
+
+const ARR_LEN: usize = Tt::const_val::<[i8; 123]>();
+//~^ ERROR constant contains unimplemented expression type
+
+trait Tt {
+    const fn const_val<T: Sized>() -> usize {
+    //~^ ERROR trait fns cannot be declared const
+        core::mem::size_of::<T>()
+    }
+}
+
+fn f(z: [f32; ARR_LEN]) -> [f32; ARR_LEN] {
+    z
+}
+
+fn main() {
+    let _ = f([1f32; ARR_LEN]);
+}
diff --git a/src/test/ui/issues/issue-54954.stderr b/src/test/ui/issues/issue-54954.stderr
new file mode 100644
index 00000000000..dd6389b8b60
--- /dev/null
+++ b/src/test/ui/issues/issue-54954.stderr
@@ -0,0 +1,16 @@
+error[E0379]: trait fns cannot be declared const
+  --> $DIR/issue-54954.rs:7:5
+   |
+LL |     const fn const_val<T: Sized>() -> usize {
+   |     ^^^^^ trait fns cannot be const
+
+error[E0019]: constant contains unimplemented expression type
+  --> $DIR/issue-54954.rs:3:24
+   |
+LL | const ARR_LEN: usize = Tt::const_val::<[i8; 123]>();
+   |                        ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
+
+error: aborting due to 2 previous errors
+
+Some errors occurred: E0019, E0379.
+For more information about an error, try `rustc --explain E0019`.