@@ -14,6 +14,7 @@ use rustc_middle::ty::{self, adjustment::PointerCast, Instance, InstanceDef, Ty,
14
14
use rustc_middle:: ty:: { Binder , TraitPredicate , TraitRef } ;
15
15
use rustc_mir_dataflow:: { self , Analysis } ;
16
16
use rustc_span:: { sym, Span , Symbol } ;
17
+ use rustc_trait_selection:: traits:: error_reporting:: InferCtxtExt ;
17
18
use rustc_trait_selection:: traits:: SelectionContext ;
18
19
19
20
use std:: mem;
@@ -808,15 +809,13 @@ impl<'tcx> Visitor<'tcx> for Checker<'_, 'tcx> {
808
809
}
809
810
810
811
let trait_ref = TraitRef :: from_method ( tcx, trait_id, substs) ;
811
- let obligation = Obligation :: new (
812
- ObligationCause :: dummy ( ) ,
813
- param_env,
814
- Binder :: dummy ( TraitPredicate {
815
- trait_ref,
816
- constness : ty:: BoundConstness :: NotConst ,
817
- polarity : ty:: ImplPolarity :: Positive ,
818
- } ) ,
819
- ) ;
812
+ let poly_trait_pred = Binder :: dummy ( TraitPredicate {
813
+ trait_ref,
814
+ constness : ty:: BoundConstness :: ConstIfConst ,
815
+ polarity : ty:: ImplPolarity :: Positive ,
816
+ } ) ;
817
+ let obligation =
818
+ Obligation :: new ( ObligationCause :: dummy ( ) , param_env, poly_trait_pred) ;
820
819
821
820
let implsrc = tcx. infer_ctxt ( ) . enter ( |infcx| {
822
821
let mut selcx = SelectionContext :: new ( & infcx) ;
@@ -860,15 +859,37 @@ impl<'tcx> Visitor<'tcx> for Checker<'_, 'tcx> {
860
859
// #[default_method_body_is_const], and the callee is in the same
861
860
// trait.
862
861
let callee_trait = tcx. trait_of_item ( callee) ;
863
- if callee_trait. is_some ( ) {
864
- if tcx. has_attr ( caller, sym:: default_method_body_is_const) {
865
- if tcx. trait_of_item ( caller) == callee_trait {
866
- nonconst_call_permission = true ;
867
- }
868
- }
862
+ if callee_trait. is_some ( )
863
+ && tcx. has_attr ( caller, sym:: default_method_body_is_const)
864
+ && callee_trait == tcx. trait_of_item ( caller)
865
+ // Can only call methods when it's `<Self as TheTrait>::f`.
866
+ && tcx. types . self_param == substs. type_at ( 0 )
867
+ {
868
+ nonconst_call_permission = true ;
869
869
}
870
870
871
871
if !nonconst_call_permission {
872
+ let obligation = Obligation :: new (
873
+ ObligationCause :: dummy_with_span ( * fn_span) ,
874
+ param_env,
875
+ tcx. mk_predicate (
876
+ poly_trait_pred. map_bound ( ty:: PredicateKind :: Trait ) ,
877
+ ) ,
878
+ ) ;
879
+
880
+ // improve diagnostics by showing what failed. Our requirements are stricter this time
881
+ // as we are going to error again anyways.
882
+ tcx. infer_ctxt ( ) . enter ( |infcx| {
883
+ if let Err ( e) = implsrc {
884
+ infcx. report_selection_error (
885
+ obligation. clone ( ) ,
886
+ & obligation,
887
+ & e,
888
+ false ,
889
+ ) ;
890
+ }
891
+ } ) ;
892
+
872
893
self . check_op ( ops:: FnCallNonConst {
873
894
caller,
874
895
callee,
0 commit comments