blob: f8e06b901ec014b75dab20a6ccdd195bbb9026c5 (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
|
use crate::coherence::CoherenceError;
use crate::ext::GoalExt;
use crate::solve::Solver;
use crate::RustIrDatabase;
use chalk_ir::cast::*;
use chalk_ir::interner::Interner;
use chalk_ir::*;
use tracing::{debug, instrument};
// Test if a local impl violates the orphan rules.
//
// For `impl<T> Trait for MyType<T>` we generate:
//
// forall<T> { LocalImplAllowed(MyType<T>: Trait) }
//
// This must be provable in order to pass the orphan check.
#[instrument(level = "debug", skip(db, solver))]
pub fn perform_orphan_check<I: Interner>(
db: &dyn RustIrDatabase<I>,
solver: &mut dyn Solver<I>,
impl_id: ImplId<I>,
) -> Result<(), CoherenceError<I>> {
let impl_datum = db.impl_datum(impl_id);
debug!(?impl_datum);
let impl_allowed: Goal<I> = impl_datum
.binders
.map_ref(|bound_impl| {
// Ignoring the polarization of the impl's polarized trait ref
DomainGoal::LocalImplAllowed(bound_impl.trait_ref.clone())
})
.cast(db.interner());
let canonical_goal = &impl_allowed.into_closed_goal(db.interner());
let is_allowed = solver.solve(db, canonical_goal).is_some();
debug!("overlaps = {:?}", is_allowed);
if !is_allowed {
let trait_id = impl_datum.trait_id();
return Err(CoherenceError::FailedOrphanCheck(trait_id));
}
Ok(())
}
|