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

cvc5-cvc5-1.2.0.src.theory.sets.theory_sets_private.cpp Maven / Gradle / Ivy

The newest version!
/******************************************************************************
 * Top contributors (to current version):
 *   Andrew Reynolds, Mudathir Mohamed, Aina Niemetz
 *
 * This file is part of the cvc5 project.
 *
 * Copyright (c) 2009-2024 by the authors listed in the file AUTHORS
 * in the top-level source directory and their institutional affiliations.
 * All rights reserved.  See the file COPYING in the top-level source
 * directory for licensing information.
 * ****************************************************************************
 *
 * Sets theory implementation.
 */

#include "theory/sets/theory_sets_private.h"

#include 
#include 

#include "expr/emptyset.h"
#include "expr/node_algorithm.h"
#include "expr/skolem_manager.h"
#include "options/quantifiers_options.h"
#include "options/sets_options.h"
#include "theory/datatypes/project_op.h"
#include "theory/datatypes/tuple_utils.h"
#include "theory/sets/normal_form.h"
#include "theory/sets/theory_sets.h"
#include "theory/theory_model.h"
#include "util/rational.h"
#include "util/result.h"

using namespace std;
using namespace cvc5::internal::kind;
using namespace cvc5::internal::theory::datatypes;

namespace cvc5::internal {
namespace theory {
namespace sets {

TheorySetsPrivate::TheorySetsPrivate(Env& env,
                                     TheorySets& external,
                                     SolverState& state,
                                     InferenceManager& im,
                                     SkolemCache& skc,
                                     CarePairArgumentCallback& cpacb)
    : EnvObj(env),
      d_deq(context()),
      d_termProcessed(userContext()),
      d_fullCheckIncomplete(false),
      d_fullCheckIncompleteId(IncompleteId::UNKNOWN),
      d_external(external),
      d_state(state),
      d_im(im),
      d_skCache(skc),
      d_treg(d_env, state, im, skc),
      d_rels(new TheorySetsRels(d_env, state, im, skc, d_treg)),
      d_cardSolver(new CardinalityExtension(d_env, state, im, d_treg)),
      d_rels_enabled(false),
      d_card_enabled(false),
      d_higher_order_kinds_enabled(false),
      d_cpacb(cpacb)
{
  d_true = nodeManager()->mkConst(true);
  d_false = nodeManager()->mkConst(false);
  d_zero = nodeManager()->mkConstInt(Rational(0));
}

TheorySetsPrivate::~TheorySetsPrivate()
{
  for (std::pair& current_pair : d_eqc_info)
  {
    delete current_pair.second;
  }
}

void TheorySetsPrivate::finishInit()
{
  d_equalityEngine = d_external.getEqualityEngine();
  Assert(d_equalityEngine != nullptr);
}

void TheorySetsPrivate::eqNotifyNewClass(TNode t)
{
  if (t.getKind() == Kind::SET_SINGLETON || t.getKind() == Kind::SET_EMPTY)
  {
    EqcInfo* e = getOrMakeEqcInfo(t, true);
    e->d_singleton = t;
  }
}

void TheorySetsPrivate::eqNotifyMerge(TNode t1, TNode t2)
{
  if (!d_state.isInConflict() && t1.getType().isSet())
  {
    Trace("sets-prop-debug")
        << "Merge " << t1 << " and " << t2 << "..." << std::endl;
    Node s1, s2;
    EqcInfo* e2 = getOrMakeEqcInfo(t2);
    if (e2)
    {
      s2 = e2->d_singleton;
      EqcInfo* e1 = getOrMakeEqcInfo(t1);
      Trace("sets-prop-debug") << "Merging singletons..." << std::endl;
      if (e1)
      {
        s1 = e1->d_singleton;
        if (!s1.isNull() && !s2.isNull())
        {
          if (s1.getKind() == s2.getKind())
          {
            Trace("sets-prop") << "Propagate eq inference : " << s1
                               << " == " << s2 << std::endl;
            // infer equality between elements of singleton
            Node exp = s1.eqNode(s2);
            Node eq = s1[0].eqNode(s2[0]);
            d_im.assertSetsFact(eq, true, InferenceId::SETS_SINGLETON_EQ, exp);
          }
          else
          {
            // singleton equal to emptyset, conflict
            Trace("sets-prop")
                << "Propagate conflict : " << s1 << " == " << s2 << std::endl;
            Node eqs = s1.eqNode(s2);
            d_im.assertSetsConflict(eqs, InferenceId::SETS_EQ_CONFLICT);
            return;
          }
        }
      }
      else
      {
        // copy information
        e1 = getOrMakeEqcInfo(t1, true);
        e1->d_singleton.set(e2->d_singleton);
      }
    }
    // merge membership list
    Trace("sets-prop-debug") << "Copying membership list..." << std::endl;
    // if s1 has a singleton or empty set and s2 does not, we may have new
    // inferences to process.
    Node checkSingleton = s2.isNull() ? s1 : Node::null();
    std::vector facts;
    // merge the membership list in the state, which may produce facts or
    // conflicts to propagate
    if (!d_state.merge(t1, t2, facts, checkSingleton))
    {
      // conflict case
      Assert(facts.size() == 1);
      Trace("sets-prop") << "Propagate eq-mem conflict : " << facts[0]
                         << std::endl;
      d_im.assertSetsConflict(facts[0], InferenceId::SETS_EQ_MEM_CONFLICT);
      return;
    }
    for (const Node& f : facts)
    {
      Assert(f.getKind() == Kind::IMPLIES);
      Trace("sets-prop") << "Propagate eq-mem eq inference : " << f[0] << " => "
                         << f[1] << std::endl;
      d_im.assertSetsFact(f[1], true, InferenceId::SETS_EQ_MEM, f[0]);
    }
  }
}

void TheorySetsPrivate::eqNotifyDisequal(TNode t1, TNode t2, TNode reason)
{
  if (t1.getType().isSet())
  {
    Node eq = t1.eqNode(t2);
    if (d_deq.find(eq) == d_deq.end())
    {
      d_deq[eq] = true;
    }
  }
}

TheorySetsPrivate::EqcInfo::EqcInfo(context::Context* c) : d_singleton(c) {}

TheorySetsPrivate::EqcInfo* TheorySetsPrivate::getOrMakeEqcInfo(TNode n,
                                                                bool doMake)
{
  std::map::iterator eqc_i = d_eqc_info.find(n);
  if (eqc_i == d_eqc_info.end())
  {
    EqcInfo* ei = NULL;
    if (doMake)
    {
      ei = new EqcInfo(context());
      d_eqc_info[n] = ei;
    }
    return ei;
  }
  else
  {
    return eqc_i->second;
  }
}

void TheorySetsPrivate::fullEffortReset()
{
  Assert(d_equalityEngine->consistent());
  d_fullCheckIncomplete = false;
  d_fullCheckIncompleteId = IncompleteId::UNKNOWN;
  d_higher_order_kinds_enabled = false;
  d_card_enabled = false;
  d_rels_enabled = false;
  // reset the state object
  d_state.reset();
  // reset the inference manager
  d_im.reset();
  d_im.clearPendingLemmas();
  // reset the cardinality solver
  d_cardSolver->reset();
}

void TheorySetsPrivate::fullEffortCheck()
{
  Trace("sets") << "----- Full effort check ------" << std::endl;
  // get the asserted terms
  std::set irrKinds;
  std::set rlvTerms;
  d_external.collectAssertedTerms(rlvTerms, true, irrKinds);
  d_external.computeRelevantTerms(rlvTerms);
  do
  {
    Assert(!d_im.hasPendingLemma() || d_im.hasSent());

    Trace("sets") << "...iterate full effort check..." << std::endl;
    fullEffortReset();

    if (TraceIsOn("sets-eqc"))
    {
      Trace("sets-eqc") << "Equality Engine:" << std::endl;
      Trace("sets-eqc") << d_equalityEngine->debugPrintEqc() << std::endl;
    }
    std::map eqcTypeCount;
    eq::EqClassesIterator eqcs_i = eq::EqClassesIterator(d_equalityEngine);
    while (!eqcs_i.isFinished())
    {
      Node eqc = (*eqcs_i);
      TypeNode tn = eqc.getType();
      d_state.registerEqc(tn, eqc);
      eqcTypeCount[tn]++;
      eq::EqClassIterator eqc_i = eq::EqClassIterator(eqc, d_equalityEngine);
      while (!eqc_i.isFinished())
      {
        Node n = (*eqc_i);
        ++eqc_i;
        // if it is not relevant, don't register it
        if (rlvTerms.find(n)==rlvTerms.end())
        {
          continue;
        }
        TypeNode tnn = n.getType();
        // register it with the state
        d_state.registerTerm(eqc, tnn, n);
        Kind nk = n.getKind();
        if (nk == Kind::SET_SINGLETON)
        {
          // ensure the proxy has been introduced
          d_treg.getProxy(n);
        }
        else if (nk == Kind::SET_CARD)
        {
          d_card_enabled = true;
          // register it with the cardinality solver
          d_cardSolver->registerTerm(n);
          if (d_im.hasSent())
          {
            break;
          }
          // if we do not handle the kind, set incomplete
          Kind nk0 = n[0].getKind();
          // some kinds of cardinality we cannot handle
          if (d_rels->isRelationKind(nk0))
          {
            d_fullCheckIncomplete = true;
            d_fullCheckIncompleteId = IncompleteId::SETS_RELS_CARD;
            Trace("sets-incomplete")
                << "Sets : incomplete because of " << n << "." << std::endl;
            // TODO (#1124):  The issue can be divided into 4 parts
            // 1- Supporting the universe cardinality for finite types with
            // finite cardinality (done)
            // 2- Supporting the universe cardinality for uninterpreted sorts
            // with finite-model-find (pending) See the implementation of
            //    CardinalityExtension::checkCardinalityExtended
            // 3- Supporting the universe cardinality for non-finite types
            // (done)
            // 4- Supporting cardinality for relations (hard)
          }
        }
        else if (d_rels->isRelationKind(nk))
        {
          d_rels_enabled = true;
        }
        else if(isHigherOrderKind(nk))
        {
          d_higher_order_kinds_enabled = true;
        }
      }
      ++eqcs_i;
    }

    if (TraceIsOn("sets-state"))
    {
      Trace("sets-state") << "Equivalence class counters:" << std::endl;
      for (std::pair& ec : eqcTypeCount)
      {
        Trace("sets-state")
            << "  " << ec.first << " -> " << ec.second << std::endl;
      }
    }

    // sources of incompleteness
    if (d_card_enabled)
    {
      if (d_higher_order_kinds_enabled)
      {
        d_fullCheckIncomplete = true;
        d_fullCheckIncompleteId = IncompleteId::SETS_HO_CARD;
      }
      if (options().quantifiers.fmfBound)
      {
        // fmfBound is incomplete since cardinality may introduce slack
        // elements.
        d_fullCheckIncomplete = true;
        d_fullCheckIncompleteId = IncompleteId::SETS_FMF_BOUND_CARD;
      }
    }

    // We may have sent lemmas while registering the terms in the loop above,
    // e.g. the cardinality solver.
    if (d_im.hasSent())
    {
      continue;
    }
    if (TraceIsOn("sets-mem"))
    {
      const std::vector& sec = d_state.getSetsEqClasses();
      for (const Node& s : sec)
      {
        Trace("sets-mem") << "Eqc " << s << " : ";
        const std::map& smem = d_state.getMembers(s);
        if (!smem.empty())
        {
          Trace("sets-mem") << "Memberships : ";
          for (const std::pair& it2 : smem)
          {
            Trace("sets-mem") << it2.first << " ";
          }
        }
        Node ss = d_state.getSingletonEqClass(s);
        if (!ss.isNull())
        {
          Trace("sets-mem") << " : Singleton : " << ss;
        }
        Trace("sets-mem") << std::endl;
      }
    }
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check downwards closure
    checkDownwardsClosure();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check upwards closure
    checkUpwardsClosure();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check filter up rule
    checkFilterUp();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check filter down rules
    checkFilterDown();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check map up rules
    checkMapUp();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check map down rules
    checkMapDown();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check group up
    checkGroups();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check disequalities
    checkDisequalities();
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    // check reduce comprehensions
    checkReduceComprehensions();

    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      continue;
    }
    if (d_card_enabled)
    {
      // call the check method of the cardinality solver
      d_cardSolver->check();
      if (d_im.hasSent())
      {
        continue;
      }
    }
    if (d_rels_enabled)
    {
      // call the check method of the relations solver
      d_rels->check(Theory::EFFORT_FULL);
    }
  } while (!d_im.hasSentLemma() && !d_state.isInConflict()
           && d_im.hasSentFact());
  Assert(!d_im.hasPendingLemma() || d_im.hasSent());
  Trace("sets") << "----- End full effort check, conflict="
                << d_state.isInConflict() << ", lemma=" << d_im.hasSentLemma()
                << std::endl;
}

void TheorySetsPrivate::checkDownwardsClosure()
{
  Trace("sets") << "TheorySetsPrivate: check downwards closure..." << std::endl;
  // downwards closure
  const std::vector& sec = d_state.getSetsEqClasses();
  for (const Node& s : sec)
  {
    const std::vector& nvsets = d_state.getNonVariableSets(s);
    if (!nvsets.empty())
    {
      const std::map& smem = d_state.getMembers(s);
      for (const Node& nv : nvsets)
      {
        for (const std::pair& it2 : smem)
        {
          Node mem = it2.second;
          Node eq_set = nv;
          Assert(d_equalityEngine->areEqual(mem[1], eq_set));
          if (mem[1] != eq_set)
          {
            Trace("sets-debug") << "Downwards closure based on " << mem
                                << ", eq_set = " << eq_set << std::endl;
            if (!options().sets.setsProxyLemmas)
            {
              Node nmem =
                  nodeManager()->mkNode(Kind::SET_MEMBER, mem[0], eq_set);
              nmem = rewrite(nmem);
              std::vector exp;
              exp.push_back(mem);
              exp.push_back(mem[1].eqNode(eq_set));
              d_im.assertInference(nmem, InferenceId::SETS_DOWN_CLOSURE, exp);
              if (d_state.isInConflict())
              {
                return;
              }
            }
            else
            {
              // use proxy set
              Node k = d_treg.getProxy(eq_set);
              Node pmem = nodeManager()->mkNode(Kind::SET_MEMBER, mem[0], k);
              Node nmem =
                  nodeManager()->mkNode(Kind::SET_MEMBER, mem[0], eq_set);
              nmem = rewrite(nmem);
              std::vector exp;
              exp.push_back(pmem);
              int inferType = 0;
              if (!d_state.areEqual(mem, pmem))
              {
                // force sending as a lemma
                inferType = 1;
              }
              d_im.assertInference(
                  nmem, InferenceId::SETS_DOWN_CLOSURE, exp, inferType);
            }
          }
        }
        
      }
    }
  }
}

void TheorySetsPrivate::checkUpwardsClosure()
{
  // upwards closure
  NodeManager* nm = nodeManager();
  const std::map > >& boi =
      d_state.getBinaryOpIndex();
  for (const std::pair > >&
           itb : boi)
  {
    Kind k = itb.first;
    Trace("sets") << "TheorySetsPrivate: check upwards closure " << k << "..."
                  << std::endl;
    for (const std::pair >& it : itb.second)
    {
      Node r1 = d_state.getRepresentative(it.first);
      // see if there are members in first argument r1
      const std::map& r1mem = d_state.getMembers(r1);
      if (!r1mem.empty() || k == Kind::SET_UNION)
      {
        for (const std::pair& it2 : it.second)
        {
          Node r2 = d_state.getRepresentative(it2.first);
          Node term = it2.second;
          // see if there are members in second argument
          const std::map& r2mem = d_state.getMembers(r2);
          const std::map& r2nmem = d_state.getNegativeMembers(r2);
          if (!r2mem.empty() || k != Kind::SET_INTER)
          {
            Trace("sets-debug")
                << "Checking " << term << ", members = " << (!r1mem.empty())
                << ", " << (!r2mem.empty()) << std::endl;
            // for all members of r1
            if (!r1mem.empty())
            {
              for (const std::pair& itm1m : r1mem)
              {
                Node xr = itm1m.first;
                Node x = itm1m.second[0];
                Trace("sets-debug") << "checking membership " << xr << " "
                                    << itm1m.second << std::endl;
                std::vector exp;
                exp.push_back(itm1m.second);
                d_state.addEqualityToExp(term[0], itm1m.second[1], exp);
                bool valid = false;
                int inferType = 0;
                if (k == Kind::SET_UNION)
                {
                  valid = true;
                }
                else if (k == Kind::SET_INTER)
                {
                  // conclude x is in term
                  // if also existing in members of r2
                  std::map::const_iterator itm = r2mem.find(xr);
                  if (itm != r2mem.end())
                  {
                    exp.push_back(itm->second);
                    d_state.addEqualityToExp(term[1], itm->second[1], exp);
                    d_state.addEqualityToExp(x, itm->second[0], exp);
                    valid = true;
                  }
                  else
                  {
                    // if not, check whether it is definitely not a member, if
                    // unknown, split
                    if (r2nmem.find(xr) == r2nmem.end())
                    {
                      exp.push_back(nm->mkNode(Kind::SET_MEMBER, x, term[1]));
                      valid = true;
                      inferType = 1;
                    }
                  }
                }
                else
                {
                  Assert(k == Kind::SET_MINUS);
                  std::map::const_iterator itm = r2mem.find(xr);
                  if (itm == r2mem.end())
                  {
                    // must add lemma for set minus since non-membership in this
                    // case is not explained
                    exp.push_back(
                        nm->mkNode(Kind::SET_MEMBER, x, term[1]).negate());
                    valid = true;
                    inferType = 1;
                  }
                }
                if (valid)
                {
                  Node rr = d_equalityEngine->getRepresentative(term);
                  if (!d_state.isMember(x, rr))
                  {
                    Node kk = d_treg.getProxy(term);
                    Node fact = nm->mkNode(Kind::SET_MEMBER, x, kk);
                    d_im.assertInference(
                        fact, InferenceId::SETS_UP_CLOSURE, exp, inferType);
                    if (d_state.isInConflict())
                    {
                      return;
                    }
                  }
                }
                Trace("sets-debug") << "done checking membership " << xr << " "
                                    << itm1m.second << std::endl;
              }
            }
            if (k == Kind::SET_UNION)
            {
              if (!r2mem.empty())
              {
                // for all members of r2
                for (const std::pair& itm2m : r2mem)
                {
                  Node x = itm2m.second[0];
                  Node rr = d_equalityEngine->getRepresentative(term);
                  if (!d_state.isMember(x, rr))
                  {
                    std::vector exp;
                    exp.push_back(itm2m.second);
                    d_state.addEqualityToExp(term[1], itm2m.second[1], exp);
                    Node r = d_treg.getProxy(term);
                    Node fact = nm->mkNode(Kind::SET_MEMBER, x, r);
                    d_im.assertInference(fact, InferenceId::SETS_UP_CLOSURE_2, exp);
                    if (d_state.isInConflict())
                    {
                      return;
                    }
                  }
                }
              }
            }
          }
        }
      }
    }
  }
  if (!d_im.hasSent())
  {
    if (options().sets.setsExt)
    {
      // universal sets
      Trace("sets-debug") << "Check universe sets..." << std::endl;
      // all elements must be in universal set
      const std::vector& sec = d_state.getSetsEqClasses();
      for (const Node& s : sec)
      {
        // if equivalence class contains a variable
        Node v = d_state.getVariableSet(s);
        if (!v.isNull())
        {
          // the variable in the equivalence class
          std::map univ_set;
          const std::map& smems = d_state.getMembers(s);
          for (const std::pair& it2 : smems)
          {
            Node e = it2.second[0];
            TypeNode tn = nodeManager()->mkSetType(e.getType());
            Node u;
            std::map::iterator itu = univ_set.find(tn);
            if (itu == univ_set.end())
            {
              Node ueqc = d_state.getUnivSetEqClass(tn);
              // if the universe does not yet exist, or is not in this
              // equivalence class
              if (s != ueqc)
              {
                u = d_treg.getUnivSet(tn);
              }
              univ_set[tn] = u;
            }
            else
            {
              u = itu->second;
            }
            if (!u.isNull())
            {
              Assert(it2.second.getKind() == Kind::SET_MEMBER);
              std::vector exp;
              exp.push_back(it2.second);
              if (v != it2.second[1])
              {
                exp.push_back(v.eqNode(it2.second[1]));
              }
              Node fact = nm->mkNode(Kind::SET_MEMBER, it2.second[0], u);
              d_im.assertInference(fact, InferenceId::SETS_UP_UNIV, exp);
              if (d_state.isInConflict())
              {
                return;
              }
            }
          }
        }
      }
    }
  }
}

void TheorySetsPrivate::checkFilterUp()
{
  NodeManager* nm = nodeManager();
  const std::vector& filterTerms = d_state.getFilterTerms();

  for (const Node& term : filterTerms)
  {
    Node p = term[0];
    Node A = term[1];
    const std::map& positiveMembers =
        d_state.getMembers(d_state.getRepresentative(A));
    for (const std::pair& pair : positiveMembers)
    {
      Node x = pair.first;
      std::vector exp;
      exp.push_back(pair.second);
      Node B = pair.second[1];
      d_state.addEqualityToExp(A, B, exp);
      Node p_x = nm->mkNode(Kind::APPLY_UF, p, x);
      Node skolem = d_treg.getProxy(term);
      Node memberFilter = nm->mkNode(Kind::SET_MEMBER, x, skolem);
      Node not_p_x = p_x.notNode();
      Node not_memberFilter = memberFilter.notNode();
      Node orNode =
          p_x.andNode(memberFilter).orNode(not_p_x.andNode(not_memberFilter));
      d_im.assertInference(orNode, InferenceId::SETS_FILTER_UP, exp);
      if (d_state.isInConflict())
      {
        return;
      }
    }
  }
}

void TheorySetsPrivate::checkFilterDown()
{
  NodeManager* nm = nodeManager();
  const std::vector& filterTerms = d_state.getFilterTerms();
  for (const Node& term : filterTerms)
  {
    Node p = term[0];
    Node A = term[1];

    const std::map& positiveMembers =
        d_state.getMembers(d_state.getRepresentative(term));
    for (const std::pair& pair : positiveMembers)
    {
      std::vector exp;
      Node B = pair.second[1];
      exp.push_back(pair.second);
      d_state.addEqualityToExp(B, term, exp);
      Node x = pair.first;
      Node memberA = nm->mkNode(Kind::SET_MEMBER, x, A);
      Node p_x = nm->mkNode(Kind::APPLY_UF, p, x);
      Node fact = memberA.andNode(p_x);
      d_im.assertInference(fact, InferenceId::SETS_FILTER_DOWN, exp);
      if (d_state.isInConflict())
      {
        return;
      }
    }
  }
}

void TheorySetsPrivate::checkMapUp()
{
  NodeManager* nm = nodeManager();
  const context::CDHashSet& mapTerms = d_state.getMapTerms();

  for (const Node& term : mapTerms)
  {
    Node f = term[0];
    Node A = term[1];
    const std::map& positiveMembers =
        d_state.getMembers(d_state.getRepresentative(A));
    shared_ptr> skolemElements =
        d_state.getMapSkolemElements(term);
    for (const std::pair& pair : positiveMembers)
    {
      Node x = pair.first;
      if (skolemElements->contains(x))
      {
        // Break this cycle between inferences SETS_MAP_DOWN_POSITIVE
        // and SETS_MAP_UP:
        // 1- If (set.member y (set.map f A)) holds, then SETS_MAP_DOWN_POSITIVE
        //    inference would generate a fresh skolem x1 such that (= (f x1) y)
        //    and (set.member x1 A).
        // 2- Since (set.member x1 A) holds, SETS_MAP_UP would infer
        //    (set.member (f x1) (set.map f A)).
        // Since (set.member (f x1) (set.map f A)) holds, step 1 would repeat
        // and generate a new skolem x2 such that (= (f x2) (f x1)) and
        // (set.member x1 A). The cycle continues with step 2.
        continue;
      }
      // (=>
      //   (and (set.member x B) (= A B))
      //   (set.member (f x) (set.map f A))
      // )
      std::vector exp;
      exp.push_back(pair.second);
      Node B = pair.second[1];
      d_state.addEqualityToExp(A, B, exp);
      Node f_x = nm->mkNode(Kind::APPLY_UF, f, x);
      Node skolem = d_treg.getProxy(term);
      Node memberMap = nm->mkNode(Kind::SET_MEMBER, f_x, skolem);
      d_im.assertInference(memberMap, InferenceId::SETS_MAP_UP, exp);
      if (d_state.isInConflict())
      {
        return;
      }
    }
  }
}

void TheorySetsPrivate::checkMapDown()
{
  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  const context::CDHashSet& mapTerms = d_state.getMapTerms();
  for (const Node& term : mapTerms)
  {
    Node f = term[0];
    Node A = term[1];
    TypeNode elementType = A.getType().getSetElementType();
    const std::map& positiveMembers =
        d_state.getMembers(d_state.getRepresentative(term));
    for (const std::pair& pair : positiveMembers)
    {
      std::vector exp;
      Node B = pair.second[1];
      exp.push_back(pair.second);
      d_state.addEqualityToExp(B, term, exp);
      Node y = pair.first;

      // general case
      // (=>
      //   (and
      //     (set.member y B)
      //     (= B (set.map f A)))
      //   (and
      //     (set.member x A)
      //     (= (f x) y))
      // )
      Node x = sm->mkSkolemFunction(SkolemId::SETS_MAP_DOWN_ELEMENT, {term, y});

      d_state.registerMapSkolemElement(term, x);
      Node memberA = nm->mkNode(Kind::SET_MEMBER, x, A);
      Node f_x = nm->mkNode(Kind::APPLY_UF, f, x);
      Node equal = f_x.eqNode(y);
      Node fact = memberA.andNode(equal);
      d_im.assertInference(fact, InferenceId::SETS_MAP_DOWN_POSITIVE, exp);
      if (d_state.isInConflict())
      {
        return;
      }
    }
  }
}

void TheorySetsPrivate::checkGroups()
{
  const context::CDHashSet& groupTerms = d_state.getGroupTerms();
  for (const Node& n : groupTerms)
  {
    checkGroup(n);
  }
}

void TheorySetsPrivate::checkGroup(Node n)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  groupNotEmpty(n);
  d_im.doPendingLemmas();
  if (d_im.hasSent())
  {
    return;
  }
  Node part = defineSkolemPartFunction(n);
  Node A = d_state.getRepresentative(n[0]);

  const std::map& membersA = d_state.getMembers(A);
  const std::map& negativeMembersA = d_state.getNegativeMembers(A);
  std::shared_ptr> skolems =
      d_state.getPartElementSkolems(n);
  for (const auto& aPair : membersA)
  {
    if (skolems->contains(aPair.first))
    {
      // skip skolem elements that were introduced by groupPartCount below.
      continue;
    }
    Node aRep = d_state.getRepresentative(aPair.first);
    groupUp1(n, aRep, part);
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      return;
    }
  }
  for (const auto& aPair : negativeMembersA)
  {
    Node aRep = d_state.getRepresentative(aPair.first);
    groupUp2(n, aRep, part);
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      return;
    }
  }
  Node nRep = d_state.getRepresentative(n);
  const std::map& parts = d_state.getMembers(nRep);
  for (std::map::const_iterator partIt1 = parts.begin();
       partIt1 != parts.end();
       ++partIt1)
  {
    Node part1 = d_state.getRepresentative(partIt1->first);
    std::vector partEqc;
    d_state.getEquivalenceClass(part1, partEqc);
    bool newPart = true;
    for (Node p : partEqc)
    {
      if (p.getKind() == Kind::APPLY_UF && p.getOperator() == part)
      {
        newPart = false;
      }
    }
    if (newPart)
    {
      // only apply the groupPartCount rule for a part that does not have
      // nodes of the form (part x) introduced by the group up rule above.
      groupPartMember(n, part1, part);
      d_im.doPendingLemmas();
      if (d_im.hasSent())
      {
        return;
      }
    }
    Node part1Rep = d_state.getRepresentative(part1);
    const std::map& partElements = d_state.getMembers(part1Rep);
    for (std::map::const_iterator i = partElements.begin();
         i != partElements.end();
         ++i)
    {
      Node x = d_state.getRepresentative(i->first);
      if (!skolems->contains(x))
      {
        // only apply down rules for elements not generated by groupPartCount
        // rule above
        groupDown(n, part1, x, part);
        d_im.doPendingLemmas();
        if (d_im.hasSent())
        {
          return;
        }
      }

      std::map::const_iterator j = i;
      ++j;
      while (j != partElements.end())
      {
        Node y = d_state.getRepresentative(j->first);
        // x, y should have the same projection
        groupSameProjection(n, part1, x, y, part);
        d_im.doPendingLemmas();
        if (d_im.hasSent())
        {
          return;
        }
        ++j;
      }

      for (const auto& aPair : membersA)
      {
        Node y = d_state.getRepresentative(aPair.first);
        if (x != y)
        {
          // x, y should have the same projection
          groupSamePart(n, part1, x, y, part);
          d_im.doPendingLemmas();
          if (d_im.hasSent())
          {
            return;
          }
        }
      }
    }
  }
}

void TheorySetsPrivate::groupNotEmpty(Node n)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  NodeManager* nm = nodeManager();
  TypeNode bagType = n.getType();
  Node A = n[0];
  Node emptyPart = nm->mkConst(EmptySet(A.getType()));
  Node skolem = registerAndAssertSkolemLemma(n);

  Node A_isEmpty = A.eqNode(emptyPart);
  std::vector exp;
  exp.push_back(A_isEmpty);
  Node singleton = nm->mkNode(Kind::SET_SINGLETON, emptyPart);
  Node groupIsSingleton = skolem.eqNode(singleton);

  Node conclusion = groupIsSingleton;
  d_im.assertInference(
      conclusion, InferenceId::SETS_RELS_GROUP_NOT_EMPTY, exp, 1);
}

void TheorySetsPrivate::groupUp1(Node n, Node x, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(x.getType() == n[0].getType().getSetElementType());
  NodeManager* nm = nodeManager();

  Node A = n[0];
  TypeNode setType = A.getType();

  Node member_x_A = nm->mkNode(Kind::SET_MEMBER, x, A);

  std::vector exp;
  exp.push_back(member_x_A);

  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);

  Node member_x_part_x = nm->mkNode(Kind::SET_MEMBER, x, part_x);

  Node skolem = registerAndAssertSkolemLemma(n);
  Node member_part_x_n = nm->mkNode(Kind::SET_MEMBER, part_x, skolem);

  Node emptyPart = nm->mkConst(EmptySet(setType));
  Node member_emptyPart = nm->mkNode(Kind::SET_MEMBER, emptyPart, skolem);
  Node emptyPart_not_member = member_emptyPart.notNode();

  Node conclusion = nm->mkNode(
      Kind::AND, {member_part_x_n, member_x_part_x, emptyPart_not_member});
  d_im.assertInference(conclusion, InferenceId::SETS_RELS_GROUP_UP1, exp, 1);
}

void TheorySetsPrivate::groupUp2(Node n, Node x, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(x.getType() == n[0].getType().getSetElementType());
  NodeManager* nm = nodeManager();
  Node A = n[0];
  TypeNode setType = A.getType();

  Node member_x_A = nm->mkNode(Kind::SET_MEMBER, x, A);

  std::vector exp;
  exp.push_back(member_x_A.notNode());

  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);
  Node part_x_is_empty = part_x.eqNode(nm->mkConst(EmptySet(setType)));
  Node conclusion = part_x_is_empty;
  d_im.assertInference(conclusion, InferenceId::SETS_RELS_GROUP_UP2, exp, 1);
}

void TheorySetsPrivate::groupDown(Node n, Node B, Node x, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(B.getType() == n.getType().getSetElementType());
  Assert(x.getType() == n[0].getType().getSetElementType());
  NodeManager* nm = nodeManager();
  Node A = n[0];
  TypeNode setType = A.getType();

  Node member_x_B = nm->mkNode(Kind::SET_MEMBER, x, B);

  Node skolem = registerAndAssertSkolemLemma(n);
  Node member_B_n = nm->mkNode(Kind::SET_MEMBER, B, skolem);
  std::vector exp;
  exp.push_back(member_B_n);
  exp.push_back(member_x_B);
  Node member_x_A = nm->mkNode(Kind::SET_MEMBER, x, A);
  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);
  Node part_x_is_B = part_x.eqNode(B);
  Node conclusion = nm->mkNode(Kind::AND, member_x_A, part_x_is_B);
  d_im.assertInference(conclusion, InferenceId::SETS_RELS_GROUP_DOWN, exp, 1);
}

void TheorySetsPrivate::groupPartMember(Node n, Node B, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(B.getType() == n.getType().getSetElementType());

  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();

  Node A = n[0];
  TypeNode setType = A.getType();
  Node empty = nm->mkConst(EmptySet(setType));

  Node skolem = registerAndAssertSkolemLemma(n);
  Node member_B_n = nm->mkNode(Kind::SET_MEMBER, B, skolem);
  std::vector exp;
  exp.push_back(member_B_n);
  Node A_notEmpty = A.eqNode(empty).notNode();
  exp.push_back(A_notEmpty);

  Node x = sm->mkSkolemFunction(SkolemId::RELATIONS_GROUP_PART_ELEMENT, {n, B});
  d_state.registerPartElementSkolem(n, x);
  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);
  Node B_is_part_x = B.eqNode(part_x);
  Node member_x_A = nm->mkNode(Kind::SET_MEMBER, x, A);
  Node member_x_B = nm->mkNode(Kind::SET_MEMBER, x, B);

  Node conclusion =
      nm->mkNode(Kind::AND, {B_is_part_x, member_x_B, member_x_A});
  d_im.assertInference(
      conclusion, InferenceId::SETS_RELS_GROUP_PART_MEMBER, exp, 1);
}

void TheorySetsPrivate::groupSameProjection(
    Node n, Node B, Node x, Node y, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(B.getType() == n.getType().getSetElementType());
  Assert(x.getType() == n[0].getType().getSetElementType());
  Assert(y.getType() == n[0].getType().getSetElementType());
  NodeManager* nm = nodeManager();

  Node A = n[0];
  TypeNode setType = A.getType();

  Node member_x_B = nm->mkNode(Kind::SET_MEMBER, x, B);
  Node member_y_B = nm->mkNode(Kind::SET_MEMBER, y, B);

  Node skolem = registerAndAssertSkolemLemma(n);
  Node member_B_n = nm->mkNode(Kind::SET_MEMBER, B, skolem);

  // premises
  std::vector exp;
  exp.push_back(member_B_n);
  exp.push_back(member_x_B);
  exp.push_back(member_y_B);
  exp.push_back(x.eqNode(y).notNode());

  const std::vector& indices =
      n.getOperator().getConst().getIndices();

  Node xProjection = TupleUtils::getTupleProjection(indices, x);
  Node yProjection = TupleUtils::getTupleProjection(indices, y);
  Node sameProjection = xProjection.eqNode(yProjection);
  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);
  Node part_y = nm->mkNode(Kind::APPLY_UF, part, y);
  part_y = registerAndAssertSkolemLemma(part_y);
  Node samePart = part_x.eqNode(part_y);
  Node part_x_is_B = part_x.eqNode(B);
  Node conclusion =
      nm->mkNode(Kind::AND, sameProjection, samePart, part_x_is_B);
  d_im.assertInference(
      conclusion, InferenceId::SETS_RELS_GROUP_SAME_PROJECTION, exp, 1);
}

void TheorySetsPrivate::groupSamePart(Node n, Node B, Node x, Node y, Node part)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Assert(B.getType() == n.getType().getSetElementType());
  Assert(x.getType() == n[0].getType().getSetElementType());
  Assert(y.getType() == n[0].getType().getSetElementType());
  NodeManager* nm = nodeManager();
  Node A = n[0];
  TypeNode setType = A.getType();

  std::vector exp;
  Node member_x_B = nm->mkNode(Kind::SET_MEMBER, x, B);
  Node member_y_A = nm->mkNode(Kind::SET_MEMBER, y, A);
  Node member_y_B = nm->mkNode(Kind::SET_MEMBER, y, B);

  Node skolem = registerAndAssertSkolemLemma(n);
  Node member_B_n = nm->mkNode(Kind::SET_MEMBER, B, skolem);
  const std::vector& indices =
      n.getOperator().getConst().getIndices();

  Node xProjection = TupleUtils::getTupleProjection(indices, x);
  Node yProjection = TupleUtils::getTupleProjection(indices, y);

  // premises
  exp.push_back(member_B_n);
  exp.push_back(member_x_B);
  exp.push_back(member_y_A);
  exp.push_back(x.eqNode(y).notNode());
  exp.push_back(xProjection.eqNode(yProjection));

  Node part_x = nm->mkNode(Kind::APPLY_UF, part, x);
  part_x = registerAndAssertSkolemLemma(part_x);
  Node part_y = nm->mkNode(Kind::APPLY_UF, part, y);
  part_y = registerAndAssertSkolemLemma(part_y);
  Node samePart = part_x.eqNode(part_y);
  Node part_x_is_B = part_x.eqNode(B);
  Node conclusion = nm->mkNode(Kind::AND, member_y_B, samePart, part_x_is_B);

  d_im.assertInference(
      conclusion, InferenceId::SETS_RELS_GROUP_SAME_PART, exp, 1);
}

Node TheorySetsPrivate::defineSkolemPartFunction(Node n)
{
  Assert(n.getKind() == Kind::RELATION_GROUP);
  Node A = n[0];

  // declare an uninterpreted function part: T -> (Relation T)
  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  Node part = sm->mkSkolemFunction(SkolemId::RELATIONS_GROUP_PART, {n});
  return part;
}

Node TheorySetsPrivate::registerAndAssertSkolemLemma(Node& n)
{
  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  Node skolem = sm->mkPurifySkolem(n);
  Node lemma = n.eqNode(skolem);
  d_im.assertInference(lemma, InferenceId::SETS_SKOLEM, d_true, 1);
  Trace("sets-skolems") << "sets-skolems:  " << skolem << " = " << n
                        << std::endl;
  return skolem;
}

void TheorySetsPrivate::checkDisequalities()
{
  // disequalities
  Trace("sets") << "TheorySetsPrivate: check disequalities..." << std::endl;
  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  for (NodeBoolMap::const_iterator it = d_deq.begin(); it != d_deq.end(); ++it)
  {
    if (!(*it).second)
    {
      // not active
      continue;
    }
    Node deq = (*it).first;
    // check if it is already satisfied
    Assert(d_equalityEngine->hasTerm(deq[0])
           && d_equalityEngine->hasTerm(deq[1]));
    Node r1 = d_equalityEngine->getRepresentative(deq[0]);
    Node r2 = d_equalityEngine->getRepresentative(deq[1]);
    bool is_sat = d_state.isSetDisequalityEntailed(r1, r2);
    Trace("sets-debug") << "Check disequality " << deq
                        << ", is_sat = " << is_sat << std::endl;
    // will process regardless of sat/processed/unprocessed
    d_deq[deq] = false;

    if (is_sat)
    {
      // already satisfied
      continue;
    }
    if (d_termProcessed.find(deq) != d_termProcessed.end())
    {
      // already added lemma
      continue;
    }
    d_termProcessed.insert(deq);
    d_termProcessed.insert(deq[1].eqNode(deq[0]));
    Trace("sets") << "Process Disequality : " << deq.negate() << std::endl;
    Node x = sm->mkSkolemFunction(SkolemId::SETS_DEQ_DIFF, {deq[0], deq[1]});
    Node mem1 = nm->mkNode(Kind::SET_MEMBER, x, deq[0]);
    Node mem2 = nm->mkNode(Kind::SET_MEMBER, x, deq[1]);
    Node mdeq = nm->mkNode(Kind::EQUAL, mem1, mem2).negate();
    d_im.assertInference(mdeq, InferenceId::SETS_DEQ, deq.notNode(), 1);
    d_im.doPendingLemmas();
    if (d_im.hasSent())
    {
      return;
    }
  }
}

void TheorySetsPrivate::checkReduceComprehensions()
{
  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  const std::vector& comps = d_state.getComprehensionSets();
  for (const Node& n : comps)
  {
    if (d_termProcessed.find(n) != d_termProcessed.end())
    {
      // already reduced it
      continue;
    }
    d_termProcessed.insert(n);
    Node v = nm->mkBoundVar(n[2].getType());
    Node body = nm->mkNode(Kind::AND, n[1], v.eqNode(n[2]));
    // must do substitution
    std::vector vars;
    std::vector subs;
    for (const Node& cv : n[0])
    {
      vars.push_back(cv);
      Node cvs = nm->mkBoundVar(cv.getType());
      subs.push_back(cvs);
    }
    body = body.substitute(vars.begin(), vars.end(), subs.begin(), subs.end());
    Node bvl = nm->mkNode(Kind::BOUND_VAR_LIST, subs);
    body = nm->mkNode(Kind::EXISTS, bvl, body);
    Node k = sm->mkPurifySkolem(n);
    Node mem = nm->mkNode(Kind::SET_MEMBER, v, k);
    Node lem = nm->mkNode(Kind::AND,
                          k.eqNode(n),
                          nm->mkNode(Kind::FORALL,
                                     nm->mkNode(Kind::BOUND_VAR_LIST, v),
                                     body.eqNode(mem)));
    Trace("sets-comprehension")
        << "Comprehension reduction: " << lem << std::endl;
    d_im.lemma(lem, InferenceId::SETS_COMPREHENSION);
  }
}

//--------------------------------- standard check

void TheorySetsPrivate::postCheck(Theory::Effort level)
{
  Trace("sets-check") << "Sets finished assertions effort " << level
                      << std::endl;
  // invoke full effort check, relations check
  if (!d_state.isInConflict())
  {
    if (level == Theory::EFFORT_FULL)
    {
      if (!d_external.d_valuation.needCheck())
      {
        fullEffortCheck();
        if (!d_state.isInConflict() && !d_im.hasSentLemma()
            && d_fullCheckIncomplete)
        {
          d_im.setModelUnsound(d_fullCheckIncompleteId);
        }
      }
    }
  }
  Trace("sets-check") << "Sets finish Check effort " << level << std::endl;
}

void TheorySetsPrivate::notifyFact(TNode atom, bool polarity, TNode fact)
{
  if (d_state.isInConflict())
  {
    return;
  }
  if (atom.getKind() == Kind::SET_MEMBER && polarity)
  {
    // check if set has a value, if so, we can propagate
    Node r = d_equalityEngine->getRepresentative(atom[1]);
    EqcInfo* e = getOrMakeEqcInfo(r, true);
    if (e)
    {
      Node s = e->d_singleton;
      if (!s.isNull())
      {
        Node pexp = nodeManager()->mkNode(Kind::AND, atom, atom[1].eqNode(s));
        if (s.getKind() == Kind::SET_SINGLETON)
        {
          if (s[0] != atom[0])
          {
            Trace("sets-prop") << "Propagate mem-eq : " << pexp << std::endl;
            Node eq = s[0].eqNode(atom[0]);
            // triggers an internal inference
            d_im.assertSetsFact(eq, true, InferenceId::SETS_MEM_EQ, pexp);
          }
        }
        else
        {
          Trace("sets-prop")
              << "Propagate mem-eq conflict : " << pexp << std::endl;
          d_im.assertSetsConflict(pexp, InferenceId::SETS_MEM_EQ_CONFLICT);
        }
      }
    }
    // add to membership list
    d_state.addMember(r, atom);
  }
}
//--------------------------------- end standard check

void TheorySetsPrivate::computeCareGraph()
{
  const std::map >& ol = d_state.getOperatorList();
  for (const std::pair >& it : ol)
  {
    Kind k = it.first;
    if (k == Kind::SET_SINGLETON || k == Kind::SET_MEMBER)
    {
      Trace("sets-cg-summary") << "Compute graph for sets, op=" << k << "..."
                               << it.second.size() << std::endl;
      Trace("sets-cg") << "Build index for " << k << "..." << std::endl;
      std::map index;
      unsigned arity = 0;
      // populate indices
      for (TNode f1 : it.second)
      {
        Trace("sets-cg-debug") << "...build for " << f1 << std::endl;
        Assert(d_equalityEngine->hasTerm(f1));
        // break into index based on operator, and the type of the element
        // type of the proper set, which notice must be safe wrt subtyping.
        TypeNode tn;
        if (k == Kind::SET_SINGLETON)
        {
          // get the type of the singleton set (not the type of its element)
          tn = f1.getType().getSetElementType();
        }
        else
        {
          Assert(k == Kind::SET_MEMBER);
          // get the element type of the set (not the type of the element)
          tn = f1[1].getType().getSetElementType();
        }
        std::vector reps;
        bool hasCareArg = false;
        for (unsigned j = 0; j < f1.getNumChildren(); j++)
        {
          reps.push_back(d_equalityEngine->getRepresentative(f1[j]));
          if (isCareArg(f1, j))
          {
            hasCareArg = true;
          }
        }
        if (hasCareArg)
        {
          Trace("sets-cg-debug") << "......adding." << std::endl;
          index[tn].addTerm(f1, reps);
          arity = reps.size();
        }
        else
        {
          Trace("sets-cg-debug") << "......skip." << std::endl;
        }
      }
      if (arity > 0)
      {
        // for each index
        for (std::pair& tt : index)
        {
          Trace("sets-cg") << "Process index " << tt.first << "..."
                           << std::endl;
          nodeTriePathPairProcess(&tt.second, arity, d_cpacb);
        }
      }
      Trace("sets-cg-summary") << "...done" << std::endl;
    }
  }
}

bool TheorySetsPrivate::isCareArg(Node n, unsigned a)
{
  if (d_equalityEngine->isTriggerTerm(n[a], THEORY_SETS))
  {
    return true;
  }
  else if ((n.getKind() == Kind::SET_MEMBER
            || n.getKind() == Kind::SET_SINGLETON)
           && a == 0 && n[0].getType().isSet())
  {
    // when the elements themselves are sets
    return true;
  }
  return false;
}

bool TheorySetsPrivate::collectModelValues(TheoryModel* m,
                                           const std::set& termSet)
{
  Trace("sets-model") << "Set collect model values" << std::endl;
  Trace("sets-model") << "termSet: " << termSet << std::endl;
  if(TraceIsOn("sets-model"))
  {
    Trace("sets-model") <debugPrintModelEqc();
  }
  NodeManager* nm = NodeManager::currentNM();
  std::map mvals;
  // If cardinality is enabled, we need to use the ordered equivalence class
  // list computed by the cardinality solver, where sets equivalence classes
  // are assigned model values based on their position in the cardinality
  // graph.
  const std::vector& sec = d_card_enabled
                                     ? d_cardSolver->getOrderedSetsEqClasses()
                                     : d_state.getSetsEqClasses();
  Valuation& val = getValuation();
  for (int i = (int)(sec.size() - 1); i >= 0; i--)
  {
    Node eqc = sec[i];
    if (termSet.find(eqc) == termSet.end())
    {
      Trace("sets-model") << "* Do not assign value for " << eqc
                          << " since is not relevant." << std::endl;
    }
    else
    {
      std::vector els;
      bool is_base = !d_card_enabled || d_cardSolver->isModelValueBasic(eqc);
      if (is_base)
      {
        Trace("sets-model")
            << "Collect elements of base eqc " << eqc << std::endl;
        // members that must be in eqc
        const std::map& emems = d_state.getMembers(eqc);
        if (!emems.empty())
        {
          for (const std::pair& itmm : emems)
          {
            // when we have y -> (set.member x S) where rep(x)=y, we use x
            // in the model here. Using y may not be legal with respect to
            // subtyping, since y may be real where x is an int.
            Node t = nm->mkNode(Kind::SET_SINGLETON, itmm.second[0]);
            els.push_back(t);
          }
        }
      }
      if (d_card_enabled)
      {
        // make the slack elements for the equivalence class based on set
        // cardinality
        d_cardSolver->mkModelValueElementsFor(val, eqc, els, mvals, m);
      }

      Node rep = NormalForm::mkBop(Kind::SET_UNION, els, eqc.getType());
      rep = rewrite(rep);
      Trace("sets-model") << "* Assign representative of " << eqc << " to "
                          << rep << std::endl;
      mvals[eqc] = rep;
      if (!m->assertEquality(eqc, rep, true))
      {
        return false;
      }
      m->assertSkeleton(rep);

      // we add the element terms (singletons) as representatives to tell the
      // model builder to evaluate them along with their union (rep).
      // This is needed to account for cases when members and rep are not enough
      // for the model builder to evaluate set terms.
      // e.g.
      // eqc(rep) = [(union (singleton skolem) (singleton 0))]
      // eqc(skolem) = [0]
      // The model builder would fail to evaluate rep as (singleton 0)
      // if (singleton skolem) is not registered as a representative in the
      // model
      for (const Node& el : els)
      {
        m->assertSkeleton(el);
      }

      Trace("sets-model") << "Set " << eqc << " = " << els << std::endl;
    }
  }

  // handle slack elements constraints for finite types
  if (d_card_enabled)
  {
    const std::map >& slackElements =
        d_cardSolver->getFiniteTypeSlackElements();
    for (const auto& pair : slackElements)
    {
      const std::vector& members =
          d_cardSolver->getFiniteTypeMembers(pair.first);
      m->setAssignmentExclusionSetGroup(pair.second, members);
      Trace("sets-model") << "ExclusionSet: Group " << pair.second
                          << " is excluded from set" << members << std::endl;
    }
  }
  return true;
}

/********************** Helper functions ***************************/
/********************** Helper functions ***************************/
/********************** Helper functions ***************************/

Valuation& TheorySetsPrivate::getValuation() { return d_external.d_valuation; }

bool TheorySetsPrivate::isEntailed(Node n, bool pol)
{
  return d_state.isEntailed(n, pol);
}

void TheorySetsPrivate::processCarePairArgs(TNode a, TNode b)
{
  for (size_t k = 0, nchild = a.getNumChildren(); k < nchild; ++k)
  {
    TNode x = a[k];
    TNode y = b[k];
    if (!d_equalityEngine->areEqual(x, y))
    {
      if (isCareArg(a, k) && isCareArg(b, k))
      {
        // splitting on sets (necessary for handling set of sets properly)
        if (x.getType().isSet())
        {
          Assert(y.getType().isSet());
          Trace("sets-cg-lemma")
              << "Should split on : " << x << "==" << y << std::endl;
          d_im.split(x.eqNode(y), InferenceId::SETS_CG_SPLIT);
        }
      }
    }
  }
}

bool TheorySetsPrivate::isHigherOrderKind(Kind k)
{
  return k == Kind::SET_MAP || k == Kind::SET_FILTER || k == Kind::SET_FOLD;
}

void TheorySetsPrivate::preRegisterTerm(TNode node)
{
  Trace("sets") << "TheorySetsPrivate::preRegisterTerm(" << node << ")"
                << std::endl;
  TypeNode tn = node.getType();
  if (tn.isSet())
  {
    ensureFirstClassSetType(tn);
  }
  switch (node.getKind())
  {
    case Kind::EQUAL:
    case Kind::SET_MEMBER:
    {
      // add trigger predicate for equality and membership
      d_state.addEqualityEngineTriggerPredicate(node);
    }
    break;
    case Kind::RELATION_JOIN_IMAGE:
    {
      // these are logic exceptions, not type checking exceptions
      if (!node[1].isConst())
      {
        throw LogicException(
            "JoinImage cardinality constraint must be a constant");
      }
      cvc5::internal::Rational r(INT_MAX);
      if (node[1].getConst() > r)
      {
        throw LogicException(
            "JoinImage Exceeded INT_MAX in cardinality constraint");
      }
      if (node[1].getConst().getNumerator().getSignedInt() < 0)
      {
        throw LogicException(
            "JoinImage cardinality constraint must be non-negative");
      }
    }
    break;
    default: d_equalityEngine->addTerm(node); break;
  }
}

TrustNode TheorySetsPrivate::ppRewrite(Node node,
                                       std::vector& lems)
{
  Trace("sets-proc") << "ppRewrite : " << node << std::endl;

  switch (node.getKind())
  {
    case Kind::SET_CHOOSE: return expandChooseOperator(node, lems);
    case Kind::SET_IS_SINGLETON: return expandIsSingletonOperator(node);
    default: break;
  }
  return TrustNode::null();
}

TrustNode TheorySetsPrivate::expandChooseOperator(
    const Node& node, std::vector& lems)
{
  Assert(node.getKind() == Kind::SET_CHOOSE);

  // (choose A) is eliminated to k, with lemma
  //   (and (= k (uf A)) (or (= A (as set.empty (Set E))) (set.member k A)))
  // where uf: (Set E) -> E is a skolem function, and E is the type of elements
  // of A

  NodeManager* nm = nodeManager();
  SkolemManager* sm = nm->getSkolemManager();
  Node x = sm->mkPurifySkolem(node);
  Node A = node[0];
  TypeNode setType = A.getType();
  ensureFirstClassSetType(setType);
  // use canonical constant to ensure it can be typed
  Node mkElem = nm->mkGroundValue(setType);
  // a Null node is used here to get a unique skolem function per set type
  Node uf = sm->mkSkolemFunction(SkolemId::SETS_CHOOSE, mkElem);
  Node ufA = nodeManager()->mkNode(Kind::APPLY_UF, uf, A);

  Node equal = x.eqNode(ufA);
  Node emptySet = nm->mkConst(EmptySet(setType));
  Node isEmpty = A.eqNode(emptySet);
  Node member = nm->mkNode(Kind::SET_MEMBER, x, A);
  Node lem =
      nm->mkNode(Kind::AND, equal, nm->mkNode(Kind::OR, isEmpty, member));
  TrustNode tlem = TrustNode::mkTrustLemma(lem, nullptr);
  lems.push_back(SkolemLemma(tlem, x));
  return TrustNode::mkTrustRewrite(node, x, nullptr);
}

TrustNode TheorySetsPrivate::expandIsSingletonOperator(const Node& node)
{
  Assert(node.getKind() == Kind::SET_IS_SINGLETON);
  Assert(rewrite(node) == node);

  // (is_singleton A) is expanded as (= A (set.singleton (set.choose A)))

  NodeManager* nm = nodeManager();
  Node choose = nm->mkNode(Kind::SET_CHOOSE, node[0]);
  Node ss = nm->mkNode(Kind::SET_SINGLETON, choose);
  Node ret = nm->mkNode(Kind::EQUAL, node[0], ss);
  return TrustNode::mkTrustRewrite(node, ret, nullptr);
}

void TheorySetsPrivate::ensureFirstClassSetType(TypeNode tn) const
{
  Assert(tn.isSet());
  if (!tn.getSetElementType().isFirstClass())
  {
    std::stringstream ss;
    ss << "Cannot handle sets of non-first class types, offending set type is "
       << tn;
    throw LogicException(ss.str());
  }
}

void TheorySetsPrivate::presolve() { d_state.reset(); }

}  // namespace sets
}  // namespace theory
}  // namespace cvc5::internal




© 2015 - 2024 Weber Informatics LLC | Privacy Policy