7627: infer: update resolver when descending into block r=jonas-schievink a=jonas-schievink



Co-authored-by: Jonas Schievink <jonasschievink@gmail.com>
This commit is contained in:
bors[bot] 2021-02-10 15:59:49 +00:00 committed by GitHub
commit 7ec03439a8
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
5 changed files with 139 additions and 32 deletions

View file

@ -1495,6 +1495,20 @@ fn main(f: Foo) {
); );
} }
#[test]
fn internal_or() {
check_diagnostics(
r#"
fn main() {
enum Either { A(bool), B }
match Either::B {
//^^^^^^^^^ Missing match arm
Either::A(true | false) => (),
}
}
"#,
);
}
mod false_negatives { mod false_negatives {
//! The implementation of match checking here is a work in progress. As we roll this out, we //! The implementation of match checking here is a work in progress. As we roll this out, we
//! prefer false negatives to false positives (ideally there would be no false positives). This //! prefer false negatives to false positives (ideally there would be no false positives). This
@ -1518,21 +1532,6 @@ fn main() {
11..20 => (), 11..20 => (),
} }
} }
"#,
);
}
#[test]
fn internal_or() {
// We do not currently handle patterns with internal `or`s.
check_diagnostics(
r#"
fn main() {
enum Either { A(bool), B }
match Either::B {
Either::A(true | false) => (),
}
}
"#, "#,
); );
} }

View file

@ -137,24 +137,33 @@ impl<'a> InferenceContext<'a> {
self.coerce_merge_branch(&then_ty, &else_ty) self.coerce_merge_branch(&then_ty, &else_ty)
} }
Expr::Block { statements, tail, label, id: _ } => match label { Expr::Block { statements, tail, label, id: _ } => {
Some(_) => { let old_resolver = mem::replace(
let break_ty = self.table.new_type_var(); &mut self.resolver,
self.breakables.push(BreakableContext { resolver_for_expr(self.db.upcast(), self.owner, tgt_expr),
may_break: false, );
break_ty: break_ty.clone(), let ty = match label {
label: label.map(|label| self.body[label].name.clone()), Some(_) => {
}); let break_ty = self.table.new_type_var();
let ty = self.infer_block(statements, *tail, &Expectation::has_type(break_ty)); self.breakables.push(BreakableContext {
let ctxt = self.breakables.pop().expect("breakable stack broken"); may_break: false,
if ctxt.may_break { break_ty: break_ty.clone(),
ctxt.break_ty label: label.map(|label| self.body[label].name.clone()),
} else { });
ty let ty =
self.infer_block(statements, *tail, &Expectation::has_type(break_ty));
let ctxt = self.breakables.pop().expect("breakable stack broken");
if ctxt.may_break {
ctxt.break_ty
} else {
ty
}
} }
} None => self.infer_block(statements, *tail, expected),
None => self.infer_block(statements, *tail, expected), };
}, self.resolver = old_resolver;
ty
}
Expr::Unsafe { body } | Expr::Const { body } => self.infer_expr(*body, expected), Expr::Unsafe { body } | Expr::Const { body } => self.infer_expr(*body, expected),
Expr::TryBlock { body } => { Expr::TryBlock { body } => {
let _inner = self.infer_expr(*body, expected); let _inner = self.infer_expr(*body, expected);

View file

@ -2415,3 +2415,50 @@ fn infer_const_params() {
"#]], "#]],
); );
} }
#[test]
fn infer_inner_type() {
check_infer(
r#"
fn foo() {
struct S { field: u32 }
let s = S { field: 0 };
let f = s.field;
}
"#,
expect![[r#"
9..89 '{ ...eld; }': ()
47..48 's': S
51..65 'S { field: 0 }': S
62..63 '0': u32
75..76 'f': u32
79..80 's': S
79..86 's.field': u32
"#]],
);
}
#[test]
fn infer_nested_inner_type() {
check_infer(
r#"
fn foo() {
{
let s = S { field: 0 };
let f = s.field;
}
struct S { field: u32 }
}
"#,
expect![[r#"
9..109 '{ ...32 } }': ()
15..79 '{ ... }': ()
29..30 's': S
33..47 'S { field: 0 }': S
44..45 '0': u32
61..62 'f': u32
65..66 's': S
65..72 's.field': u32
"#]],
);
}

View file

@ -3151,3 +3151,54 @@ fn test() {
"#, "#,
); );
} }
#[test]
fn inner_use() {
check_types(
r#"
mod m {
pub trait Tr {
fn method(&self) -> u8 { 0 }
}
impl Tr for () {}
}
fn f() {
use m::Tr;
().method();
//^^^^^^^^^^^ u8
}
"#,
);
}
#[test]
fn inner_use_in_block() {
check_types(
r#"
mod m {
pub trait Tr {
fn method(&self) -> u8 { 0 }
}
impl Tr for () {}
}
fn f() {
{
use m::Tr;
().method();
//^^^^^^^^^^^ u8
}
{
().method();
//^^^^^^^^^^^ {unknown}
}
}
"#,
);
}

View file

@ -654,6 +654,7 @@ fn main() {
let test = "test"; let test = "test";
//^^^^ &str //^^^^ &str
let test = InnerStruct {}; let test = InnerStruct {};
//^^^^ InnerStruct
let test = unresolved(); let test = unresolved();