aboutsummaryrefslogtreecommitdiff
path: root/crates
diff options
context:
space:
mode:
authorMarcus Klaas de Vries <[email protected]>2019-01-07 19:11:31 +0000
committerMarcus Klaas de Vries <[email protected]>2019-01-07 19:11:31 +0000
commit7b0eaef58072acc087d23faca5a9f9879f1765d5 (patch)
tree753250bc428b0313faac6de7622686c445ef7dd6 /crates
parent3238c06a5a122b7e7b9b6871484c700b7947fae1 (diff)
Implement type inference for more binary operators
Mostly just for primitive numeric types such as u32 and f64. Not yet a general solution using trait resolution.
Diffstat (limited to 'crates')
-rw-r--r--crates/ra_hir/src/ty.rs85
-rw-r--r--crates/ra_hir/src/ty/tests.rs13
-rw-r--r--crates/ra_hir/src/ty/tests/data/boolean_op.txt31
-rw-r--r--crates/ra_syntax/src/ast.rs4
4 files changed, 84 insertions, 49 deletions
diff --git a/crates/ra_hir/src/ty.rs b/crates/ra_hir/src/ty.rs
index 23ee76f4e..e09279a68 100644
--- a/crates/ra_hir/src/ty.rs
+++ b/crates/ra_hir/src/ty.rs
@@ -529,7 +529,7 @@ struct InferenceContext<'a, D: HirDatabase> {
529 529
530// helper function that determines whether a binary operator 530// helper function that determines whether a binary operator
531// always returns a boolean 531// always returns a boolean
532fn is_boolean_operator(op: BinaryOp) -> bool { 532fn boolean_op_return_ty(op: BinaryOp, rhs_ty: Ty) -> Ty {
533 match op { 533 match op {
534 BinaryOp::BooleanOr 534 BinaryOp::BooleanOr
535 | BinaryOp::BooleanAnd 535 | BinaryOp::BooleanAnd
@@ -537,8 +537,32 @@ fn is_boolean_operator(op: BinaryOp) -> bool {
537 | BinaryOp::LesserEqualTest 537 | BinaryOp::LesserEqualTest
538 | BinaryOp::GreaterEqualTest 538 | BinaryOp::GreaterEqualTest
539 | BinaryOp::LesserTest 539 | BinaryOp::LesserTest
540 | BinaryOp::GreaterTest => true, 540 | BinaryOp::GreaterTest => Ty::Bool,
541 _ => false, 541 BinaryOp::Assignment
542 | BinaryOp::AddAssign
543 | BinaryOp::SubAssign
544 | BinaryOp::DivAssign
545 | BinaryOp::MulAssign
546 | BinaryOp::RemAssign
547 | BinaryOp::ShrAssign
548 | BinaryOp::ShlAssign
549 | BinaryOp::BitAndAssign
550 | BinaryOp::BitOrAssign
551 | BinaryOp::BitXorAssign => Ty::unit(),
552 BinaryOp::Addition
553 | BinaryOp::Subtraction
554 | BinaryOp::Multiplication
555 | BinaryOp::Division
556 | BinaryOp::Remainder
557 | BinaryOp::LeftShift
558 | BinaryOp::RightShift
559 | BinaryOp::BitwiseAnd
560 | BinaryOp::BitwiseOr
561 | BinaryOp::BitwiseXor => match rhs_ty {
562 Ty::Uint(..) | Ty::Int(..) | Ty::Float(..) => rhs_ty,
563 _ => Ty::Unknown,
564 },
565 BinaryOp::RangeRightOpen | BinaryOp::RangeRightClosed => Ty::Unknown,
542 } 566 }
543} 567}
544 568
@@ -890,20 +914,59 @@ impl<'a, D: HirDatabase> InferenceContext<'a, D> {
890 } 914 }
891 Expr::BinaryOp { lhs, rhs, op } => match op { 915 Expr::BinaryOp { lhs, rhs, op } => match op {
892 Some(op) => { 916 Some(op) => {
893 let subtype_expectation = match op { 917 let lhs_expectation = match op {
894 BinaryOp::BooleanAnd | BinaryOp::BooleanOr => { 918 BinaryOp::BooleanAnd | BinaryOp::BooleanOr => {
895 Expectation::has_type(Ty::Bool) 919 Expectation::has_type(Ty::Bool)
896 } 920 }
897 _ => Expectation::none(), 921 _ => Expectation::none(),
898 }; 922 };
899 let _lhs_ty = self.infer_expr(*lhs, &subtype_expectation)?; 923 let lhs_ty = self.infer_expr(*lhs, &lhs_expectation)?;
900 let _rhs_ty = self.infer_expr(*rhs, &subtype_expectation)?; 924 // TODO: find implementation of trait corresponding to operation
925 // symbol and resolve associated `Output` type
926 let rhs_expectation = match op {
927 BinaryOp::BooleanAnd | BinaryOp::BooleanOr => Ty::Bool,
928 BinaryOp::Assignment | BinaryOp::EqualityTest => match lhs_ty {
929 Ty::Uint(..)
930 | Ty::Int(..)
931 | Ty::Float(..)
932 | Ty::Str
933 | Ty::Char
934 | Ty::Bool => lhs_ty,
935 _ => Ty::Unknown,
936 },
937 BinaryOp::LesserEqualTest
938 | BinaryOp::GreaterEqualTest
939 | BinaryOp::LesserTest
940 | BinaryOp::GreaterTest
941 | BinaryOp::AddAssign
942 | BinaryOp::SubAssign
943 | BinaryOp::DivAssign
944 | BinaryOp::MulAssign
945 | BinaryOp::RemAssign
946 | BinaryOp::ShrAssign
947 | BinaryOp::ShlAssign
948 | BinaryOp::BitAndAssign
949 | BinaryOp::BitOrAssign
950 | BinaryOp::BitXorAssign
951 | BinaryOp::Addition
952 | BinaryOp::Subtraction
953 | BinaryOp::Multiplication
954 | BinaryOp::Division
955 | BinaryOp::Remainder
956 | BinaryOp::LeftShift
957 | BinaryOp::RightShift
958 | BinaryOp::BitwiseAnd
959 | BinaryOp::BitwiseOr
960 | BinaryOp::BitwiseXor => match lhs_ty {
961 Ty::Uint(..) | Ty::Int(..) | Ty::Float(..) => lhs_ty,
962 _ => Ty::Unknown,
963 },
964 _ => Ty::Unknown,
965 };
966 let rhs_ty = self.infer_expr(*rhs, &Expectation::has_type(rhs_expectation))?;
901 967
902 if is_boolean_operator(*op) { 968 // TODO: similar as above, return ty is often associated trait type
903 Ty::Bool 969 boolean_op_return_ty(*op, rhs_ty)
904 } else {
905 Ty::Unknown
906 }
907 } 970 }
908 _ => Ty::Unknown, 971 _ => Ty::Unknown,
909 }, 972 },
diff --git a/crates/ra_hir/src/ty/tests.rs b/crates/ra_hir/src/ty/tests.rs
index e6c7e225b..2749d740c 100644
--- a/crates/ra_hir/src/ty/tests.rs
+++ b/crates/ra_hir/src/ty/tests.rs
@@ -157,7 +157,7 @@ impl S {
157} 157}
158 158
159#[test] 159#[test]
160fn infer_boolean_op() { 160fn infer_binary_op() {
161 check_inference( 161 check_inference(
162 r#" 162 r#"
163fn f(x: bool) -> i32 { 163fn f(x: bool) -> i32 {
@@ -168,15 +168,18 @@ fn test() {
168 let x = a && b; 168 let x = a && b;
169 let y = true || false; 169 let y = true || false;
170 let z = x == y; 170 let z = x == y;
171 let h = CONST_1 <= CONST_2; 171 let minus_forty: isize = -40isize;
172 let h = minus_forty <= CONST_2;
172 let c = f(z || y) + 5; 173 let c = f(z || y) + 5;
173 let d = b; 174 let d = b;
174 let e = 3i32 && "hello world"; 175 let g = minus_forty ^= i;
176 let ten: usize = 10;
177 let ten_is_eleven = ten == some_num;
175 178
176 10 < 3 179 ten < 3
177} 180}
178"#, 181"#,
179 "boolean_op.txt", 182 "binary_op.txt",
180 ); 183 );
181} 184}
182 185
diff --git a/crates/ra_hir/src/ty/tests/data/boolean_op.txt b/crates/ra_hir/src/ty/tests/data/boolean_op.txt
deleted file mode 100644
index cce8d68fb..000000000
--- a/crates/ra_hir/src/ty/tests/data/boolean_op.txt
+++ /dev/null
@@ -1,31 +0,0 @@
1[6; 7) 'x': [unknown]
2[22; 34) '{ 0i32 }': i32
3[28; 32) '0i32': i32
4[46; 237) '{ ... < 3 }': bool
5[56; 57) 'x': bool
6[60; 61) 'a': bool
7[60; 66) 'a && b': bool
8[65; 66) 'b': bool
9[76; 77) 'y': bool
10[80; 84) 'true': bool
11[80; 93) 'true || false': bool
12[88; 93) 'false': bool
13[103; 104) 'z': bool
14[107; 108) 'x': bool
15[107; 113) 'x == y': bool
16[112; 113) 'y': bool
17[123; 124) 'h': bool
18[127; 134) 'CONST_1': [unknown]
19[127; 145) 'CONST_...ONST_2': bool
20[138; 145) 'CONST_2': [unknown]
21[155; 156) 'c': [unknown]
22[159; 172) 'f(z || y) + 5': [unknown]
23[182; 183) 'd': [unknown]
24[186; 187) 'b': [unknown]
25[197; 198) 'e': bool
26[201; 205) '3i32': bool
27[201; 222) '3i32 &...world"': bool
28[209; 222) '"hello world"': bool
29[229; 231) '10': [unknown]
30[229; 235) '10 < 3': bool
31[234; 235) '3': [unknown]
diff --git a/crates/ra_syntax/src/ast.rs b/crates/ra_syntax/src/ast.rs
index d8e187514..9ab59738f 100644
--- a/crates/ra_syntax/src/ast.rs
+++ b/crates/ra_syntax/src/ast.rs
@@ -549,7 +549,7 @@ pub enum BinOp {
549 /// The `&=` operator for assignment after bitwise AND 549 /// The `&=` operator for assignment after bitwise AND
550 BitAndAssign, 550 BitAndAssign,
551 /// The `^=` operator for assignment after bitwise XOR 551 /// The `^=` operator for assignment after bitwise XOR
552 BitXorAssin, 552 BitXorAssign,
553} 553}
554 554
555impl<'a> BinExpr<'a> { 555impl<'a> BinExpr<'a> {
@@ -586,7 +586,7 @@ impl<'a> BinExpr<'a> {
586 MINUSEQ => Some(BinOp::SubAssign), 586 MINUSEQ => Some(BinOp::SubAssign),
587 PIPEEQ => Some(BinOp::BitOrAssign), 587 PIPEEQ => Some(BinOp::BitOrAssign),
588 AMPEQ => Some(BinOp::BitAndAssign), 588 AMPEQ => Some(BinOp::BitAndAssign),
589 CARETEQ => Some(BinOp::BitXorAssin), 589 CARETEQ => Some(BinOp::BitXorAssign),
590 _ => None, 590 _ => None,
591 }) 591 })
592 .next() 592 .next()