rust/src/comp/middle/check_alt.rs

151 lines
4.4 KiB
Rust
Raw Normal View History

import syntax::ast::*;
2011-09-12 18:13:28 -05:00
import syntax::ast_util::{variant_def_ids, dummy_sp};
import syntax::visit;
fn check_crate(tcx: ty::ctxt, crate: @crate) {
2011-07-27 07:19:39 -05:00
let v =
@{visit_expr: bind check_expr(tcx, _, _, _),
visit_local: bind check_local(tcx, _, _, _)
with *visit::default_visitor::<()>()};
visit::visit_crate(*crate, (), visit::mk_vt(v));
tcx.sess.abort_if_errors();
}
fn check_expr(tcx: ty::ctxt, ex: @expr, s: (), v: visit::vt<()>) {
visit::visit_expr(ex, s, v);
2011-07-27 07:19:39 -05:00
alt ex.node { expr_alt(_, arms) { check_arms(tcx, arms); } _ { } }
}
fn check_arms(tcx: ty::ctxt, arms: [arm]) {
2011-07-27 07:19:39 -05:00
let i = 0;
for arm: arm in arms {
for arm_pat: @pat in arm.pats {
2011-07-27 07:19:39 -05:00
let reachable = true;
let j = 0;
while j < i {
if std::option::is_none(arms[j].guard) {
for prev_pat: @pat in arms[j].pats {
if pattern_supersedes(tcx, prev_pat, arm_pat) {
reachable = false;
}
}
}
j += 1;
}
if !reachable {
2011-09-02 17:34:58 -05:00
tcx.sess.span_err(arm_pat.span, "unreachable pattern");
}
}
i += 1;
}
}
fn pattern_supersedes(tcx: ty::ctxt, a: @pat, b: @pat) -> bool {
fn patterns_supersede(tcx: ty::ctxt, as: [@pat], bs: [@pat]) -> bool {
2011-07-27 07:19:39 -05:00
let i = 0;
for a: @pat in as {
if !pattern_supersedes(tcx, a, bs[i]) { ret false; }
i += 1;
}
ret true;
}
fn field_patterns_supersede(tcx: ty::ctxt, fas: [field_pat],
fbs: [field_pat]) -> bool {
let wild = @{id: 0, node: pat_wild, span: dummy_sp()};
for fa: field_pat in fas {
2011-07-27 07:19:39 -05:00
let pb = wild;
for fb: field_pat in fbs {
if fa.ident == fb.ident { pb = fb.pat; }
}
if !pattern_supersedes(tcx, fa.pat, pb) { ret false; }
}
ret true;
}
alt a.node {
2011-07-27 07:19:39 -05:00
pat_wild. | pat_bind(_) { ret true; }
pat_lit(la) {
alt b.node {
2011-07-27 07:19:39 -05:00
pat_lit(lb) { ret util::common::lit_eq(la, lb); }
_ { ret false; }
}
}
2011-07-27 07:19:39 -05:00
pat_tag(va, suba) {
alt b.node {
2011-07-27 07:19:39 -05:00
pat_tag(vb, subb) {
ret tcx.def_map.get(a.id) == tcx.def_map.get(b.id) &&
2011-07-27 07:19:39 -05:00
patterns_supersede(tcx, suba, subb);
}
_ { ret false; }
}
}
2011-07-27 07:19:39 -05:00
pat_rec(suba, _) {
alt b.node {
2011-07-27 07:19:39 -05:00
pat_rec(subb, _) { ret field_patterns_supersede(tcx, suba, subb); }
_ { ret false; }
}
}
2011-08-15 06:15:19 -05:00
pat_tup(suba) {
alt b.node {
pat_tup(subb) { ret patterns_supersede(tcx, suba, subb); }
_ { ret false; }
}
}
2011-07-27 07:19:39 -05:00
pat_box(suba) {
alt b.node {
2011-07-27 07:19:39 -05:00
pat_box(subb) { ret pattern_supersedes(tcx, suba, subb); }
_ { ret pattern_supersedes(tcx, suba, b); }
}
}
pat_uniq(suba) {
alt b.node {
pat_uniq(subb) { ret pattern_supersedes(tcx, suba, subb); }
_ { ret pattern_supersedes(tcx, suba, b); }
}
}
}
}
fn check_local(tcx: ty::ctxt, loc: @local, s: (), v: visit::vt<()>) {
visit::visit_local(loc, s, v);
if is_refutable(tcx, loc.node.pat) {
tcx.sess.span_err(loc.node.pat.span,
2011-09-02 17:34:58 -05:00
"refutable pattern in local binding");
}
}
fn is_refutable(tcx: ty::ctxt, pat: @pat) -> bool {
alt pat.node {
pat_wild. | pat_bind(_) { ret false; }
pat_lit(_) { ret true; }
pat_box(sub) { ret is_refutable(tcx, sub); }
pat_uniq(sub) { ret is_refutable(tcx, sub); }
pat_rec(fields, _) {
for field: field_pat in fields {
if is_refutable(tcx, field.pat) { ret true; }
}
ret false;
}
2011-08-15 06:15:19 -05:00
pat_tup(elts) {
for elt in elts { if is_refutable(tcx, elt) { ret true; } }
2011-08-15 06:15:19 -05:00
ret false;
}
pat_tag(_, args) {
let vdef = variant_def_ids(tcx.def_map.get(pat.id));
if std::vec::len(ty::tag_variants(tcx, vdef.tg)) != 1u { ret true; }
for p: @pat in args { if is_refutable(tcx, p) { ret true; } }
ret false;
}
}
}
// Local Variables:
// mode: rust
// fill-column: 78;
// indent-tabs-mode: nil
// c-basic-offset: 4
// buffer-file-coding-system: utf-8-unix
// compile-command: "make -k -C $RBUILD 2>&1 | sed -e 's/\\/x\\//x:\\//g'";
// End: