11
11
#![ allow( non_camel_case_types) ]
12
12
13
13
use self :: ConstVal :: * ;
14
-
15
14
use self :: ErrKind :: * ;
15
+ use self :: EvalHint :: * ;
16
16
17
17
use ast_map;
18
18
use ast_map:: blocks:: FnLikeNode ;
@@ -345,7 +345,7 @@ pub fn const_expr_to_pat(tcx: &ty::ctxt, expr: &Expr, span: Span) -> P<ast::Pat>
345
345
}
346
346
347
347
pub fn eval_const_expr ( tcx : & ty:: ctxt , e : & Expr ) -> ConstVal {
348
- match eval_const_expr_partial ( tcx, e, None ) {
348
+ match eval_const_expr_partial ( tcx, e, ExprTypeChecked ) {
349
349
Ok ( r) => r,
350
350
Err ( s) => tcx. sess . span_fatal ( s. span , & s. description ( ) )
351
351
}
@@ -434,6 +434,28 @@ impl ConstEvalErr {
434
434
pub type EvalResult = Result < ConstVal , ConstEvalErr > ;
435
435
pub type CastResult = Result < ConstVal , ErrKind > ;
436
436
437
+ // FIXME: Long-term, this enum should go away: trying to evaluate
438
+ // an expression which hasn't been type-checked is a recipe for
439
+ // disaster. That said, it's not clear how to fix ast_ty_to_ty
440
+ // to avoid the ordering issue.
441
+
442
+ /// Hint to determine how to evaluate constant expressions which
443
+ /// might not be type-checked.
444
+ #[ derive( Copy , Clone , Debug ) ]
445
+ pub enum EvalHint < ' tcx > {
446
+ /// We have a type-checked expression.
447
+ ExprTypeChecked ,
448
+ /// We have an expression which hasn't been type-checked, but we have
449
+ /// an idea of what the type will be because of the context. For example,
450
+ /// the length of an array is always `usize`. (This is referred to as
451
+ /// a hint because it isn't guaranteed to be consistent with what
452
+ /// type-checking would compute.)
453
+ UncheckedExprHint ( Ty < ' tcx > ) ,
454
+ /// We have an expression which has not yet been type-checked, and
455
+ /// and we have no clue what the type will be.
456
+ UncheckedExprNoHint ,
457
+ }
458
+
437
459
#[ derive( Copy , Clone , PartialEq , Debug ) ]
438
460
pub enum IntTy { I8 , I16 , I32 , I64 }
439
461
#[ derive( Copy , Clone , PartialEq , Debug ) ]
@@ -704,26 +726,34 @@ pub_fn_checked_op!{ const_uint_checked_shr_via_int(a: u64, b: i64,.. UintTy) {
704
726
uint_shift_body overflowing_shr Uint ShiftRightWithOverflow
705
727
} }
706
728
707
- // After type checking, `eval_const_expr_partial` should always suffice. The
708
- // reason for providing `eval_const_expr_with_substs ` is to allow
709
- // trait-associated consts to be evaluated * during* type checking, when the
710
- // substs for each expression have not been written into `tcx` yet.
729
+ /// Evaluate a constant expression in a context where the expression isn't
730
+ /// guaranteed to be evaluatable. `ty_hint ` is usually ExprTypeChecked,
731
+ /// but a few places need to evaluate constants during type- checking, like
732
+ /// computing the length of an array. (See also the FIXME above EvalHint.)
711
733
pub fn eval_const_expr_partial < ' tcx > ( tcx : & ty:: ctxt < ' tcx > ,
712
734
e : & Expr ,
713
- ty_hint : Option < Ty < ' tcx > > ) -> EvalResult {
714
- eval_const_expr_with_substs ( tcx, e, ty_hint, |id| {
715
- tcx. node_id_item_substs ( id) . substs
716
- } )
717
- }
718
-
719
- pub fn eval_const_expr_with_substs < ' tcx , S > ( tcx : & ty:: ctxt < ' tcx > ,
720
- e : & Expr ,
721
- ty_hint : Option < Ty < ' tcx > > ,
722
- get_substs : S ) -> EvalResult
723
- where S : Fn ( ast:: NodeId ) -> subst:: Substs < ' tcx > {
735
+ ty_hint : EvalHint < ' tcx > ) -> EvalResult {
724
736
fn fromb ( b : bool ) -> ConstVal { Int ( b as i64 ) }
725
737
726
- let ety = ty_hint. or_else ( || tcx. expr_ty_opt ( e) ) ;
738
+ // Try to compute the type of the expression based on the EvalHint.
739
+ // (See also the definition of EvalHint, and the FIXME above EvalHint.)
740
+ let ety = match ty_hint {
741
+ ExprTypeChecked => {
742
+ // After type-checking, expr_ty is guaranteed to succeed.
743
+ Some ( tcx. expr_ty ( e) )
744
+ }
745
+ UncheckedExprHint ( ty) => {
746
+ // Use the type hint; it's not guaranteed to be right, but it's
747
+ // usually good enough.
748
+ Some ( ty)
749
+ }
750
+ UncheckedExprNoHint => {
751
+ // This expression might not be type-checked, and we have no hint.
752
+ // Try to query the context for a type anyway; we might get lucky
753
+ // (for example, if the expression was imported from another crate).
754
+ tcx. expr_ty_opt ( e)
755
+ }
756
+ } ;
727
757
728
758
// If type of expression itself is int or uint, normalize in these
729
759
// bindings so that isize/usize is mapped to a type with an
@@ -739,7 +769,7 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
739
769
740
770
let result = match e. node {
741
771
ast:: ExprUnary ( ast:: UnNeg , ref inner) => {
742
- match try!( eval_const_expr_partial ( tcx, & * * inner, ety ) ) {
772
+ match try!( eval_const_expr_partial ( tcx, & * * inner, ty_hint ) ) {
743
773
Float ( f) => Float ( -f) ,
744
774
Int ( n) => try!( const_int_checked_neg ( n, e, expr_int_type) ) ,
745
775
Uint ( i) => {
@@ -749,7 +779,7 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
749
779
}
750
780
}
751
781
ast:: ExprUnary ( ast:: UnNot , ref inner) => {
752
- match try!( eval_const_expr_partial ( tcx, & * * inner, ety ) ) {
782
+ match try!( eval_const_expr_partial ( tcx, & * * inner, ty_hint ) ) {
753
783
Int ( i) => Int ( !i) ,
754
784
Uint ( i) => const_uint_not ( i, expr_uint_type) ,
755
785
Bool ( b) => Bool ( !b) ,
@@ -758,10 +788,16 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
758
788
}
759
789
ast:: ExprBinary ( op, ref a, ref b) => {
760
790
let b_ty = match op. node {
761
- ast:: BiShl | ast:: BiShr => Some ( tcx. types . usize ) ,
762
- _ => ety
791
+ ast:: BiShl | ast:: BiShr => {
792
+ if let ExprTypeChecked = ty_hint {
793
+ ExprTypeChecked
794
+ } else {
795
+ UncheckedExprHint ( tcx. types . usize )
796
+ }
797
+ }
798
+ _ => ty_hint
763
799
} ;
764
- match ( try!( eval_const_expr_partial ( tcx, & * * a, ety ) ) ,
800
+ match ( try!( eval_const_expr_partial ( tcx, & * * a, ty_hint ) ) ,
765
801
try!( eval_const_expr_partial ( tcx, & * * b, b_ty) ) ) {
766
802
( Float ( a) , Float ( b) ) => {
767
803
match op. node {
@@ -851,22 +887,25 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
851
887
}
852
888
}
853
889
ast:: ExprCast ( ref base, ref target_ty) => {
854
- // This tends to get called w/o the type actually having been
855
- // populated in the ctxt, which was causing things to blow up
856
- // (#5900). Fall back to doing a limited lookup to get past it.
857
890
let ety = ety. or_else ( || ast_ty_to_prim_ty ( tcx, & * * target_ty) )
858
891
. unwrap_or_else ( || {
859
892
tcx. sess . span_fatal ( target_ty. span ,
860
893
"target type not found for const cast" )
861
894
} ) ;
862
895
863
- // Prefer known type to noop, but always have a type hint.
864
- //
865
- // FIXME (#23833): the type-hint can cause problems,
866
- // e.g. `(i8::MAX + 1_i8) as u32` feeds in `u32` as result
867
- // type to the sum, and thus no overflow is signaled.
868
- let base_hint = tcx. expr_ty_opt ( & * * base) . unwrap_or ( ety) ;
869
- let val = try!( eval_const_expr_partial ( tcx, & * * base, Some ( base_hint) ) ) ;
896
+ let base_hint = if let ExprTypeChecked = ty_hint {
897
+ ExprTypeChecked
898
+ } else {
899
+ // FIXME (#23833): the type-hint can cause problems,
900
+ // e.g. `(i8::MAX + 1_i8) as u32` feeds in `u32` as result
901
+ // type to the sum, and thus no overflow is signaled.
902
+ match tcx. expr_ty_opt ( & base) {
903
+ Some ( t) => UncheckedExprHint ( t) ,
904
+ None => ty_hint
905
+ }
906
+ } ;
907
+
908
+ let val = try!( eval_const_expr_partial ( tcx, & * * base, base_hint) ) ;
870
909
match cast_const ( tcx, val, ety) {
871
910
Ok ( val) => val,
872
911
Err ( kind) => return Err ( ConstEvalErr { span : e. span , kind : kind } ) ,
@@ -896,12 +935,16 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
896
935
def:: FromTrait ( trait_id) => match tcx. map . find ( def_id. node ) {
897
936
Some ( ast_map:: NodeTraitItem ( ti) ) => match ti. node {
898
937
ast:: ConstTraitItem ( ref ty, _) => {
899
- let substs = get_substs ( e. id ) ;
900
- ( resolve_trait_associated_const ( tcx,
901
- ti,
902
- trait_id,
903
- substs) ,
904
- Some ( & * * ty) )
938
+ if let ExprTypeChecked = ty_hint {
939
+ let substs = tcx. node_id_item_substs ( e. id ) . substs ;
940
+ ( resolve_trait_associated_const ( tcx,
941
+ ti,
942
+ trait_id,
943
+ substs) ,
944
+ Some ( & * * ty) )
945
+ } else {
946
+ ( None , None )
947
+ }
905
948
}
906
949
_ => ( None , None )
907
950
} ,
@@ -930,27 +973,42 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
930
973
Some ( actual_e) => actual_e,
931
974
None => signal ! ( e, NonConstPath )
932
975
} ;
933
- let ety = ety. or_else ( || const_ty. and_then ( |ty| ast_ty_to_prim_ty ( tcx, ty) ) ) ;
934
- try!( eval_const_expr_partial ( tcx, const_expr, ety) )
976
+ let item_hint = if let UncheckedExprNoHint = ty_hint {
977
+ match const_ty {
978
+ Some ( ty) => match ast_ty_to_prim_ty ( tcx, ty) {
979
+ Some ( ty) => UncheckedExprHint ( ty) ,
980
+ None => UncheckedExprNoHint
981
+ } ,
982
+ None => UncheckedExprNoHint
983
+ }
984
+ } else {
985
+ ty_hint
986
+ } ;
987
+ try!( eval_const_expr_partial ( tcx, const_expr, item_hint) )
935
988
}
936
989
ast:: ExprLit ( ref lit) => {
937
990
lit_to_const ( & * * lit, ety)
938
991
}
939
- ast:: ExprParen ( ref e) => try!( eval_const_expr_partial ( tcx, & * * e, ety ) ) ,
992
+ ast:: ExprParen ( ref e) => try!( eval_const_expr_partial ( tcx, & * * e, ty_hint ) ) ,
940
993
ast:: ExprBlock ( ref block) => {
941
994
match block. expr {
942
- Some ( ref expr) => try!( eval_const_expr_partial ( tcx, & * * expr, ety ) ) ,
995
+ Some ( ref expr) => try!( eval_const_expr_partial ( tcx, & * * expr, ty_hint ) ) ,
943
996
None => Int ( 0 )
944
997
}
945
998
}
946
999
ast:: ExprTup ( _) => Tuple ( e. id ) ,
947
1000
ast:: ExprStruct ( ..) => Struct ( e. id ) ,
948
1001
ast:: ExprTupField ( ref base, index) => {
949
- if let Ok ( c) = eval_const_expr_partial ( tcx, base, None ) {
1002
+ let base_hint = if let ExprTypeChecked = ty_hint {
1003
+ ExprTypeChecked
1004
+ } else {
1005
+ UncheckedExprNoHint
1006
+ } ;
1007
+ if let Ok ( c) = eval_const_expr_partial ( tcx, base, base_hint) {
950
1008
if let Tuple ( tup_id) = c {
951
1009
if let ast:: ExprTup ( ref fields) = tcx. map . expect_expr ( tup_id) . node {
952
1010
if index. node < fields. len ( ) {
953
- return eval_const_expr_partial ( tcx, & fields[ index. node ] , None )
1011
+ return eval_const_expr_partial ( tcx, & fields[ index. node ] , base_hint )
954
1012
} else {
955
1013
signal ! ( e, TupleIndexOutOfBounds ) ;
956
1014
}
@@ -966,13 +1024,18 @@ pub fn eval_const_expr_with_substs<'tcx, S>(tcx: &ty::ctxt<'tcx>,
966
1024
}
967
1025
ast:: ExprField ( ref base, field_name) => {
968
1026
// Get the base expression if it is a struct and it is constant
969
- if let Ok ( c) = eval_const_expr_partial ( tcx, base, None ) {
1027
+ let base_hint = if let ExprTypeChecked = ty_hint {
1028
+ ExprTypeChecked
1029
+ } else {
1030
+ UncheckedExprNoHint
1031
+ } ;
1032
+ if let Ok ( c) = eval_const_expr_partial ( tcx, base, base_hint) {
970
1033
if let Struct ( struct_id) = c {
971
1034
if let ast:: ExprStruct ( _, ref fields, _) = tcx. map . expect_expr ( struct_id) . node {
972
1035
// Check that the given field exists and evaluate it
973
1036
if let Some ( f) = fields. iter ( ) . find ( |f| f. ident . node . as_str ( )
974
1037
== field_name. node . as_str ( ) ) {
975
- return eval_const_expr_partial ( tcx, & * f. expr , None )
1038
+ return eval_const_expr_partial ( tcx, & * f. expr , base_hint )
976
1039
} else {
977
1040
signal ! ( e, MissingStructField ) ;
978
1041
}
@@ -1148,21 +1211,17 @@ pub fn compare_const_vals(a: &ConstVal, b: &ConstVal) -> Option<Ordering> {
1148
1211
} )
1149
1212
}
1150
1213
1151
- pub fn compare_lit_exprs < ' tcx , S > ( tcx : & ty:: ctxt < ' tcx > ,
1152
- a : & Expr ,
1153
- b : & Expr ,
1154
- ty_hint : Option < Ty < ' tcx > > ,
1155
- get_substs : S ) -> Option < Ordering >
1156
- where S : Fn ( ast:: NodeId ) -> subst:: Substs < ' tcx > {
1157
- let a = match eval_const_expr_with_substs ( tcx, a, ty_hint,
1158
- |id| { get_substs ( id) } ) {
1214
+ pub fn compare_lit_exprs < ' tcx > ( tcx : & ty:: ctxt < ' tcx > ,
1215
+ a : & Expr ,
1216
+ b : & Expr ) -> Option < Ordering > {
1217
+ let a = match eval_const_expr_partial ( tcx, a, ExprTypeChecked ) {
1159
1218
Ok ( a) => a,
1160
1219
Err ( e) => {
1161
1220
tcx. sess . span_err ( a. span , & e. description ( ) ) ;
1162
1221
return None ;
1163
1222
}
1164
1223
} ;
1165
- let b = match eval_const_expr_with_substs ( tcx, b, ty_hint , get_substs ) {
1224
+ let b = match eval_const_expr_partial ( tcx, b, ExprTypeChecked ) {
1166
1225
Ok ( b) => b,
1167
1226
Err ( e) => {
1168
1227
tcx. sess . span_err ( b. span , & e. description ( ) ) ;
0 commit comments