@@ -8,7 +8,7 @@ use rustc_errors::codes::*;
8
8
use rustc_hir:: def:: { CtorKind , DefKind } ;
9
9
use rustc_hir:: { Node , intravisit} ;
10
10
use rustc_infer:: infer:: { RegionVariableOrigin , TyCtxtInferExt } ;
11
- use rustc_infer:: traits:: Obligation ;
11
+ use rustc_infer:: traits:: { Obligation , ObligationCauseCode } ;
12
12
use rustc_lint_defs:: builtin:: {
13
13
REPR_TRANSPARENT_EXTERNAL_PRIVATE_FIELDS , UNSUPPORTED_FN_PTR_CALLING_CONVENTIONS ,
14
14
} ;
@@ -267,7 +267,12 @@ fn check_opaque_meets_bounds<'tcx>(
267
267
def_id : LocalDefId ,
268
268
origin : hir:: OpaqueTyOrigin < LocalDefId > ,
269
269
) -> Result < ( ) , ErrorGuaranteed > {
270
- let span = span_of_opaque ( tcx, def_id, origin) . unwrap_or_else ( || tcx. def_span ( def_id) ) ;
270
+ let ( span, definition_def_id) =
271
+ if let Some ( ( span, def_id) ) = best_definition_site_of_opaque ( tcx, def_id, origin) {
272
+ ( span, Some ( def_id) )
273
+ } else {
274
+ ( tcx. def_span ( def_id) , None )
275
+ } ;
271
276
272
277
let defining_use_anchor = match origin {
273
278
hir:: OpaqueTyOrigin :: FnReturn { parent, .. }
@@ -305,8 +310,32 @@ fn check_opaque_meets_bounds<'tcx>(
305
310
_ => re,
306
311
} ) ;
307
312
308
- let misc_cause = traits:: ObligationCause :: misc ( span, def_id) ;
313
+ // HACK: We eagerly instantiate some bounds to report better errors for them...
314
+ // This isn't necessary for correctness, since we register these bounds when
315
+ // equating the opaque below, but we should clean this up in the new solver.
316
+ for ( predicate, pred_span) in
317
+ tcx. explicit_item_bounds ( def_id) . iter_instantiated_copied ( tcx, args)
318
+ {
319
+ let predicate = predicate. fold_with ( & mut BottomUpFolder {
320
+ tcx,
321
+ ty_op : |ty| if ty == opaque_ty { hidden_ty } else { ty } ,
322
+ lt_op : |lt| lt,
323
+ ct_op : |ct| ct,
324
+ } ) ;
325
+
326
+ ocx. register_obligation ( Obligation :: new (
327
+ tcx,
328
+ ObligationCause :: new (
329
+ span,
330
+ def_id,
331
+ ObligationCauseCode :: OpaqueTypeBound ( pred_span, definition_def_id) ,
332
+ ) ,
333
+ param_env,
334
+ predicate,
335
+ ) ) ;
336
+ }
309
337
338
+ let misc_cause = ObligationCause :: misc ( span, def_id) ;
310
339
// FIXME: We should just register the item bounds here, rather than equating.
311
340
match ocx. eq ( & misc_cause, param_env, opaque_ty, hidden_ty) {
312
341
Ok ( ( ) ) => { }
@@ -364,33 +393,33 @@ fn check_opaque_meets_bounds<'tcx>(
364
393
}
365
394
}
366
395
367
- fn span_of_opaque < ' tcx > (
396
+ fn best_definition_site_of_opaque < ' tcx > (
368
397
tcx : TyCtxt < ' tcx > ,
369
398
opaque_def_id : LocalDefId ,
370
399
origin : hir:: OpaqueTyOrigin < LocalDefId > ,
371
- ) -> Option < Span > {
400
+ ) -> Option < ( Span , LocalDefId ) > {
372
401
struct TaitConstraintLocator < ' tcx > {
373
402
opaque_def_id : LocalDefId ,
374
403
tcx : TyCtxt < ' tcx > ,
375
404
}
376
405
impl < ' tcx > TaitConstraintLocator < ' tcx > {
377
- fn check ( & self , item_def_id : LocalDefId ) -> ControlFlow < Span > {
406
+ fn check ( & self , item_def_id : LocalDefId ) -> ControlFlow < ( Span , LocalDefId ) > {
378
407
if !self . tcx . has_typeck_results ( item_def_id) {
379
408
return ControlFlow :: Continue ( ( ) ) ;
380
409
}
381
410
382
411
if let Some ( hidden_ty) =
383
412
self . tcx . mir_borrowck ( item_def_id) . concrete_opaque_types . get ( & self . opaque_def_id )
384
413
{
385
- ControlFlow :: Break ( hidden_ty. span )
414
+ ControlFlow :: Break ( ( hidden_ty. span , item_def_id ) )
386
415
} else {
387
416
ControlFlow :: Continue ( ( ) )
388
417
}
389
418
}
390
419
}
391
420
impl < ' tcx > intravisit:: Visitor < ' tcx > for TaitConstraintLocator < ' tcx > {
392
421
type NestedFilter = nested_filter:: All ;
393
- type Result = ControlFlow < Span > ;
422
+ type Result = ControlFlow < ( Span , LocalDefId ) > ;
394
423
fn nested_visit_map ( & mut self ) -> Self :: Map {
395
424
self . tcx . hir ( )
396
425
}
0 commit comments