Skip to content

Commit 702ea71

Browse files
author
Alex Burka
committed
typeck: use NoExpectation to check return type of diverging fn
This fixes #35849, a regression introduced by the typeck refactoring around TyNever/!.
1 parent 4901896 commit 702ea71

File tree

5 files changed

+49
-2
lines changed

5 files changed

+49
-2
lines changed

src/librustc/middle/liveness.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -1479,7 +1479,13 @@ impl<'a, 'tcx> Liveness<'a, 'tcx> {
14791479
self.ir.tcx.region_maps.call_site_extent(id, body.id),
14801480
&self.fn_ret(id));
14811481

1482-
if self.live_on_entry(entry_ln, self.s.no_ret_var).is_some() {
1482+
if fn_ret.is_never() {
1483+
// FIXME(durka) this rejects code like `fn foo(x: !) -> ! { x }`
1484+
if self.live_on_entry(entry_ln, self.s.clean_exit_var).is_some() {
1485+
span_err!(self.ir.tcx.sess, sp, E0270,
1486+
"computation may converge in a function marked as diverging");
1487+
}
1488+
} else if self.live_on_entry(entry_ln, self.s.no_ret_var).is_some() {
14831489
let param_env = ParameterEnvironment::for_item(self.ir.tcx, id);
14841490
let t_ret_subst = fn_ret.subst(self.ir.tcx, &param_env.free_substs);
14851491
let is_nil = self.ir.tcx.infer_ctxt(None, Some(param_env),

src/librustc_typeck/check/mod.rs

+7-1
Original file line numberDiff line numberDiff line change
@@ -708,7 +708,13 @@ fn check_fn<'a, 'gcx, 'tcx>(inherited: &'a Inherited<'a, 'gcx, 'tcx>,
708708

709709
inherited.tables.borrow_mut().liberated_fn_sigs.insert(fn_id, fn_sig);
710710

711-
fcx.check_block_with_expected(body, ExpectHasType(fcx.ret_ty));
711+
// FIXME(aburka) do we need this special case? and should it be is_uninhabited?
712+
let expected = if fcx.ret_ty.is_never() {
713+
NoExpectation
714+
} else {
715+
ExpectHasType(fcx.ret_ty)
716+
};
717+
fcx.check_block_with_expected(body, expected);
712718

713719
fcx
714720
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,16 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
fn _converge() -> ! { //~ ERROR computation may converge
12+
42
13+
}
14+
15+
fn main() { }
16+

src/test/run-fail/call-fn-never-arg.rs

+1
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,7 @@
1010

1111
// Test that we can use a ! for an argument of type !
1212

13+
// ignore-test FIXME(durka) can't be done with the current liveness code
1314
// error-pattern:wowzers!
1415

1516
#![feature(never_type)]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// Copyright 2016 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
fn assert_sizeof() -> ! {
12+
unsafe {
13+
::std::mem::transmute::<f64, [u8; 8]>(panic!())
14+
}
15+
}
16+
17+
fn main() { }
18+

0 commit comments

Comments
 (0)