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

org.aya.util.terck.Selector Maven / Gradle / Ivy

// Copyright (c) 2020-2023 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.util.terck;

import kala.collection.SeqView;
import kala.collection.immutable.ImmutableSeq;
import kala.collection.mutable.MutableList;
import kala.tuple.Tuple;
import kala.tuple.Tuple2;
import kala.value.Var;
import org.jetbrains.annotations.NotNull;

public interface Selector {
  sealed interface Result {}

  /** The old one is better, The new one is useless. */
  record Useless(@NotNull A better) implements Result {}

  /** The new one is better, and it beats all {@link #junks}, but it still worse than {@link #betters} */
  record Evolve(@NotNull SeqView junks, @NotNull SeqView betters) implements Result {}

  static > @NotNull Result select(@NotNull A a, @NotNull SeqView had) {
    if (had.isEmpty()) return new Evolve<>(SeqView.empty(), SeqView.empty());
    var b = had.getFirst();
    var bs = had.drop(1);
    // The code below looks confusing on "what is better?".
    // In Relation class, we say `a` is better than `b` if a decreases more.
    // Here we say `a` is better than `b` if `a` contains less information (namely decrease less)
    // because we are in the context of completing a call graph which is to find the
    // worst call matrix (combining two call matrices always loses some information on argument size change)
    // and use it to check termination (because it signifies the worst case of the recursion).
    return switch (a.compare(b)) {
      // `a` is not strictly better than b, dropping `a`
      case Eq, Gt -> new Useless<>(b);
      // `a` is strictly better than b, dropping `b`
      case Lt -> switch (select(a, bs)) {
        case Evolve e -> new Evolve<>(e.junks.appended(b), e.betters);
        case Useless u -> u;
      };
      // cannot compare, keep both
      case Unk -> switch (select(a, bs)) {
        case Evolve e -> new Evolve<>(e.junks, e.betters.appended(b));
        case Useless u -> u;
      };
    };
  }

  static > @NotNull Tuple2, ImmutableSeq>
  select(@NotNull SeqView new_, @NotNull SeqView old_) {
    var old = new Var<>(old_);
    var winners = MutableList.create();
    new_.forEach(a -> {
      switch (select(a, old.value)) {
        case Evolve e -> {
          winners.append(a);
          // dropping elements worse than `a`
          old.value = e.betters;
        }
        case Useless u -> {}
      }
    });
    return Tuple.of(winners.toImmutableSeq(), old.value.toImmutableSeq());
  }

  /** A weaker {@link Relation} used in decrease amount (speed) comparison */
  enum DecrOrd {
    Lt, Eq, Gt, Unk;

    public static @NotNull DecrOrd compareBool(boolean l, boolean r) {
      if (!l && r) return DecrOrd.Lt;
      if (l && !r) return DecrOrd.Gt;
      return DecrOrd.Eq;
    }

    public static @NotNull DecrOrd compareInt(int l, int r) {
      if (l < r) return DecrOrd.Lt;
      if (l > r) return DecrOrd.Gt;
      return DecrOrd.Eq;
    }

    public @NotNull DecrOrd add(@NotNull DecrOrd rhs) {
      if (this == Unk) return rhs;

      if (this == Lt && rhs == Lt) return Lt;
      if (this == Lt && rhs == Eq) return Eq;

      if (this == Eq && rhs == Lt) return Eq;
      if (this == Eq && rhs == Eq) return Eq;
      if (this == Eq && rhs == Gt) return Eq;

      if (this == Gt && rhs == Eq) return Eq;
      if (this == Gt && rhs == Gt) return Gt;

      return Unk;
    }

    public @NotNull DecrOrd mul(@NotNull DecrOrd rhs) {
      if (this == Unk) return Unk;
      if (this == Eq) return rhs;

      if (this == Lt && rhs == Lt) return Lt;
      if (this == Lt && rhs == Eq) return Lt;

      if (this == Gt && rhs == Eq) return Gt;
      if (this == Gt && rhs == Gt) return Gt;

      return Unk;
    }
  }

  interface Candidate {
    /** Compare elements by their decrease amount. */
    @NotNull DecrOrd compare(@NotNull T other);
    default boolean notWorseThan(@NotNull T other) {
      // If `this` is not worse than `other`, `this` should decrease more or equal to `other`.
      return switch (compare(other)) {
        case Lt, Unk -> false;
        case Eq, Gt -> true;
      };
    }
  }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy