All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.aya.tyck.tycker.Unifiable 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.tyck.tycker;

import org.aya.syntax.concrete.Expr;
import org.aya.syntax.core.Closure;
import org.aya.syntax.core.term.Term;
import org.aya.syntax.core.term.xtt.DimTerm;
import org.aya.syntax.core.term.xtt.EqTerm;
import org.aya.tyck.error.UnifyError;
import org.aya.tyck.error.UnifyInfo;
import org.aya.unify.TermComparator;
import org.aya.util.Ordering;
import org.aya.util.error.SourcePos;
import org.aya.util.error.WithPos;
import org.aya.util.reporter.Problem;
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;

import java.util.function.Function;

public interface Unifiable extends Problematic, Stateful {
  @NotNull TermComparator unifier(@NotNull SourcePos pos, @NotNull Ordering order);

  /**
   * Check whether {@param lower} is a subtype of {@param upper} if {@param ordering} is {@link Ordering#Lt}.
   *
   * @param ord by default should be {@link Ordering#Lt}, or {@link Ordering#Eq}
   * @return failure data, null if success
   */
  default @Nullable TermComparator.FailureData unifyTerm(
    @NotNull Term upper, @NotNull Term lower, @Nullable Term type,
    @NotNull SourcePos pos, Ordering ord
  ) {
    var unifier = unifier(pos, ord);
    var result = unifier.compare(lower, upper, type);
    if (!result) return unifier.getFailure();
    return null;
  }

  /**
   * @param pc a problem constructor
   * @see Unifiable#unifyTerm
   */
  default boolean unifyTyReported(
    @NotNull Term upper, @NotNull Term lower, @NotNull SourcePos pos,
    @NotNull Function pc
  ) {
    var result = unifyTerm(upper, lower, null, pos, Ordering.Lt);
    if (result != null) fail(pc.apply(new UnifyInfo.Comparison(
      freezeHoles(lower), freezeHoles(upper), result)));
    return result == null;
  }

  default void checkBoundaries(
    EqTerm eq, Closure core, @NotNull SourcePos pos,
    @NotNull Function report
  ) {
    unifyTermReported(core.apply(DimTerm.I0), eq.a(), eq.appA(DimTerm.I0), pos, report);
    unifyTermReported(core.apply(DimTerm.I1), eq.b(), eq.appA(DimTerm.I1), pos, report);
  }

  default boolean unifyTermReported(
    @NotNull Term lhs, @NotNull Term rhs, @Nullable Term type, @NotNull SourcePos pos,
    @NotNull Function pc
  ) {
    var result = unifyTerm(lhs, rhs, type, pos, Ordering.Eq);
    if (result != null) fail(pc.apply(new UnifyInfo.Comparison(
      freezeHoles(rhs), freezeHoles(lhs), result)));
    return result == null;
  }

  default boolean unifyTyReported(@NotNull Term upper, @NotNull Term lower, @NotNull WithPos expr) {
    return unifyTyReported(upper, lower, expr.sourcePos(),
      cp -> new UnifyError.Type(expr.data(), expr.sourcePos(), cp, new UnifyInfo(state())));
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy