org.aya.unify.DoubleChecker Maven / Gradle / Ivy
// Copyright (c) 2020-2024 Tesla (Yinsen) Zhang.
// Use of this source code is governed by the MIT license that can be found in the LICENSE.md file.
package org.aya.unify;
import org.aya.generic.term.DTKind;
import org.aya.syntax.core.term.*;
import org.aya.syntax.core.term.call.MetaCall;
import org.aya.syntax.core.term.xtt.DimTyTerm;
import org.aya.syntax.core.term.xtt.EqTerm;
import org.aya.syntax.ref.LocalCtx;
import org.aya.syntax.ref.LocalVar;
import org.aya.syntax.ref.MetaVar;
import org.aya.tyck.TyckState;
import org.aya.tyck.error.BadExprError;
import org.aya.tyck.tycker.Contextful;
import org.aya.tyck.tycker.Problematic;
import org.aya.tyck.tycker.Stateful;
import org.aya.util.error.Panic;
import org.aya.util.reporter.Problem;
import org.aya.util.reporter.Reporter;
import org.jetbrains.annotations.NotNull;
import java.util.function.Function;
/**
* @apiNote {@link Unifier#localCtx()} should be the same object as {@link Synthesizer#localCtx()}
*/
public record DoubleChecker(
@NotNull Unifier unifier,
@NotNull Synthesizer synthesizer
) implements Stateful, Contextful, Problematic {
public DoubleChecker(@NotNull Unifier unifier) { this(unifier, new Synthesizer(unifier.nameGen, unifier)); }
public boolean inherit(@NotNull Term preterm, @NotNull Term expected) {
return switch (preterm) {
case ErrorTerm _ -> true;
case DepTypeTerm(var kind, var pParam, var pBody) -> switch (kind) {
case Pi -> {
if (!(whnf(expected) instanceof SortTerm expectedTy)) yield Panic.unreachable();
yield synthesizer.inheritPiDom(pParam, expectedTy) && subscoped(pParam, param ->
inherit(pBody.apply(param), expectedTy));
}
case Sigma -> inherit(pParam, expected) && subscoped(pParam, param ->
inherit(pBody.apply(param), expected));
};
case TupTerm(var lhs, var rhs) when whnf(expected) instanceof
DepTypeTerm(var kind, var lhsT, var rhsTClos) && kind == DTKind.Sigma ->
inherit(lhs, lhsT) && inherit(rhs, rhsTClos.apply(lhs));
case LamTerm(var body) -> switch (whnf(expected)) {
case DepTypeTerm(var kind, var dom, var cod) when kind == DTKind.Pi -> subscoped(dom, param ->
inherit(body.apply(param), cod.apply(param)));
case EqTerm eq -> subscoped(DimTyTerm.INSTANCE, param -> {
// TODO: check boundaries
return inherit(body.apply(param), eq.A().apply(param));
});
default -> failF(new BadExprError(preterm, unifier.pos, expected));
};
case TupTerm _ -> failF(new BadExprError(preterm, unifier.pos, expected));
case MetaCall(var ref, var args) when !(ref.req() instanceof MetaVar.OfType) -> {
var newMeta = new MetaCall(new MetaVar(
ref.name(), ref.pos(), ref.ctxSize(), new MetaVar.OfType(expected), false), args);
unifier.compare(preterm, newMeta, null);
yield true;
}
default -> unifier.compare(synthesizer.synthDontNormalize(preterm), expected, null);
};
}
private boolean failF(@NotNull Problem problem) {
fail(problem);
return false;
}
@Override public @NotNull LocalCtx localCtx() { return unifier.localCtx(); }
@Override public @NotNull LocalCtx setLocalCtx(@NotNull LocalCtx ctx) { return unifier.setLocalCtx(ctx); }
@Override public @NotNull TyckState state() { return unifier.state(); }
@Override public @NotNull Reporter reporter() { return unifier.reporter(); }
public R subscoped(@NotNull Term type, @NotNull Function action) {
return unifier.subscoped(type, action);
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy