Skip to content

Commit 68f8812

Browse files
committed
Guard against infinitely expanding generic/inline functions
Closes #2220 Test case disabled until a memory-leak issue is resolved.
1 parent 2782cfb commit 68f8812

File tree

3 files changed

+49
-6
lines changed

3 files changed

+49
-6
lines changed

src/rustc/middle/trans/base.rs

+18-6
Original file line numberDiff line numberDiff line change
@@ -1938,31 +1938,41 @@ fn monomorphic_fn(ccx: @crate_ctxt, fn_id: ast::def_id, real_substs: [ty::t],
19381938

19391939
let map_node = ccx.tcx.items.get(fn_id.node);
19401940
// Get the path so that we can create a symbol
1941-
let (pt, name) = alt map_node {
1941+
let (pt, name, span) = alt map_node {
19421942
ast_map::node_item(i, pt) {
19431943
alt i.node {
19441944
ast::item_res(_, _, _, dtor_id, _, _) {
19451945
item_ty = ty::node_id_to_type(ccx.tcx, dtor_id);
19461946
}
19471947
_ {}
19481948
}
1949-
(pt, i.ident)
1949+
(pt, i.ident, i.span)
19501950
}
1951-
ast_map::node_variant(v, _, pt) { (pt, v.node.name) }
1952-
ast_map::node_method(m, _, pt) { (pt, m.ident) }
1951+
ast_map::node_variant(v, enm, pt) { (pt, v.node.name, enm.span) }
1952+
ast_map::node_method(m, _, pt) { (pt, m.ident, m.span) }
19531953
ast_map::node_native_item(i, ast::native_abi_rust_intrinsic, pt)
1954-
{ (pt, i.ident) }
1954+
{ (pt, i.ident, i.span) }
19551955
ast_map::node_native_item(_, abi, _) {
19561956
// Natives don't have to be monomorphized.
19571957
ret {val: get_item_val(ccx, fn_id.node),
19581958
must_cast: true};
19591959
}
1960-
ast_map::node_ctor(nm, _, _, pt) { (pt, nm) }
1960+
ast_map::node_ctor(nm, _, _, pt) { (pt, nm, ast_util::dummy_sp()) }
19611961
_ { fail "unexpected node type"; }
19621962
};
19631963
let mono_ty = ty::subst_tps(ccx.tcx, substs, item_ty);
19641964
let llfty = type_of_fn_from_ty(ccx, mono_ty);
19651965

1966+
let depth = option::get_or_default(ccx.monomorphizing.find(fn_id), 0u);
1967+
// Random cut-off -- code that needs to instantiate the same function
1968+
// recursively more than ten times can probably safely be assumed to be
1969+
// causing an infinite expansion.
1970+
if depth > 10u {
1971+
ccx.sess.span_fatal(
1972+
span, "overly deep expansion of inlined function");
1973+
}
1974+
ccx.monomorphizing.insert(fn_id, depth + 1u);
1975+
19661976
let pt = *pt + [path_name(ccx.names(name))];
19671977
let s = mangle_exported_name(ccx, pt, mono_ty);
19681978
let lldecl = decl_internal_cdecl_fn(ccx.llmod, s, llfty);
@@ -2014,6 +2024,7 @@ fn monomorphic_fn(ccx: @crate_ctxt, fn_id: ast::def_id, real_substs: [ty::t],
20142024
}
20152025
}
20162026
}
2027+
ccx.monomorphizing.insert(fn_id, depth);
20172028
{val: lldecl, must_cast: must_cast}
20182029
}
20192030

@@ -5022,6 +5033,7 @@ fn trans_crate(sess: session::session, crate: @ast::crate, tcx: ty::ctxt,
50225033
tydescs: ty::new_ty_hash(),
50235034
external: util::common::new_def_hash(),
50245035
monomorphized: map::hashmap(hash_mono_id, {|a, b| a == b}),
5036+
monomorphizing: ast_util::new_def_id_hash(),
50255037
type_use_cache: util::common::new_def_hash(),
50265038
vtables: map::hashmap(hash_mono_id, {|a, b| a == b}),
50275039
const_cstr_cache: map::str_hash(),

src/rustc/middle/trans/common.rs

+1
Original file line numberDiff line numberDiff line change
@@ -95,6 +95,7 @@ type crate_ctxt = {
9595
external: hashmap<ast::def_id, option<ast::node_id>>,
9696
// Cache instances of monomorphized functions
9797
monomorphized: hashmap<mono_id, ValueRef>,
98+
monomorphizing: hashmap<ast::def_id, uint>,
9899
// Cache computed type parameter uses (see type_use.rs)
99100
type_use_cache: hashmap<ast::def_id, [type_use::type_uses]>,
100101
// Cache generated vtables
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
// error-pattern: overly deep expansion
2+
// issue 2258
3+
// This is currently exposing a memory leak, and xfailed for that reason
4+
// xfail-test
5+
6+
iface to_opt {
7+
fn to_option() -> option<self>;
8+
}
9+
10+
impl of to_opt for uint {
11+
fn to_option() -> option<uint> {
12+
some(self)
13+
}
14+
}
15+
16+
impl<T:copy> of to_opt for option<T> {
17+
fn to_option() -> option<option<T>> {
18+
some(self)
19+
}
20+
}
21+
22+
fn function<T:to_opt>(counter: uint, t: T) {
23+
if counter > 0u {
24+
function(counter - 1u, t.to_option());
25+
}
26+
}
27+
28+
fn main() {
29+
function(22u, 22u);
30+
}

0 commit comments

Comments
 (0)