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

cvc5-cvc5-1.2.0.include.cvc5.cvc5.h Maven / Gradle / Ivy

The newest version!
/******************************************************************************
 * Top contributors (to current version):
 *   Aina Niemetz, Andrew Reynolds, Gereon Kremer
 *
 * 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.
 * ****************************************************************************
 *
 * The cvc5 C++ API.
 */

#include 

#ifndef CVC5__API__CVC5_H
#define CVC5__API__CVC5_H

#include 
#include 
#include 
#include 

#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 

namespace cvc5 {

namespace main {
class CommandExecutor;
}  // namespace main

namespace internal {

#ifndef DOXYGEN_SKIP
template 
class NodeTemplate;
typedef NodeTemplate Node;
#endif

class DType;
class DTypeConstructor;
class DTypeSelector;
class NodeManager;
class SolverEngine;
class TypeNode;
class ProofNode;
class Options;
class Random;
class Rational;
class Result;
class SygusGrammar;
class SynthResult;
class StatisticsRegistry;
}  // namespace internal

namespace parser {
class Cmd;
}

class TermManager;
class Solver;
class Statistics;
struct APIStatistics;
class Term;
class PluginInternal;

/* -------------------------------------------------------------------------- */
/* Exception                                                                  */
/* -------------------------------------------------------------------------- */

/**
 * Base class for all API exceptions.
 * If thrown, all API objects may be in an unsafe state.
 */
class CVC5_EXPORT CVC5ApiException : public std::exception
{
 public:
  /**
   * Construct with message from a string.
   * @param str The error message.
   */
  CVC5ApiException(const std::string& str) : d_msg(str) {}
  /**
   * Construct with message from a string stream.
   * @param stream The error message.
   */
  CVC5ApiException(const std::stringstream& stream) : d_msg(stream.str()) {}
  /**
   * Retrieve the message from this exception.
   * @return The error message.
   */
  const std::string& getMessage() const { return d_msg; }
  /**
   * Retrieve the message as a C-style array.
   * @return The error message.
   */
  const char* what() const noexcept override { return d_msg.c_str(); }

  /**
   * Printing: feel free to redefine toStream().  When overridden in
   * a derived class, it's recommended that this method print the
   * type of exception before the actual message.
   */
  virtual void toStream(std::ostream& os) const { os << d_msg; }

 private:
  /** The stored error message. */
  std::string d_msg;
};

inline std::ostream& operator<<(std::ostream& os, const CVC5ApiException& e)
{
  e.toStream(os);
  return os;
}

/**
 * A recoverable API exception.
 * If thrown, API objects can still be used.
 */
class CVC5_EXPORT CVC5ApiRecoverableException : public CVC5ApiException
{
 public:
  /**
   * Construct with message from a string.
   * @param str The error message.
   */
  CVC5ApiRecoverableException(const std::string& str) : CVC5ApiException(str) {}
  /**
   * Construct with message from a string stream.
   * @param stream The error message.
   */
  CVC5ApiRecoverableException(const std::stringstream& stream)
      : CVC5ApiException(stream.str())
  {
  }
};

/**
 * Exception for unsupported command arguments.
 * If thrown, API objects can still be used.
 */
class CVC5_EXPORT CVC5ApiUnsupportedException
    : public CVC5ApiRecoverableException
{
 public:
  /**
   * Construct with message from a string.
   * @param str The error message.
   */
  CVC5ApiUnsupportedException(const std::string& str)
      : CVC5ApiRecoverableException(str)
  {
  }
  /**
   * Construct with message from a string stream.
   * @param stream The error message.
   */
  CVC5ApiUnsupportedException(const std::stringstream& stream)
      : CVC5ApiRecoverableException(stream.str())
  {
  }
};

/**
 * An option-related API exception.
 * If thrown, API objects can still be used.
 */
class CVC5_EXPORT CVC5ApiOptionException : public CVC5ApiRecoverableException
{
 public:
  /**
   * Construct with message from a string.
   * @param str The error message.
   */
  CVC5ApiOptionException(const std::string& str)
      : CVC5ApiRecoverableException(str)
  {
  }
  /**
   * Construct with message from a string stream.
   * @param stream The error message.
   */
  CVC5ApiOptionException(const std::stringstream& stream)
      : CVC5ApiRecoverableException(stream.str())
  {
  }
};

/* -------------------------------------------------------------------------- */
/* Result                                                                     */
/* -------------------------------------------------------------------------- */

/**
 * Encapsulation of a three-valued solver result, with explanations.
 */
class CVC5_EXPORT Result
{
  friend class Solver;

 public:
  /** Constructor. */
  Result();

  /**
   * Determine if this Result is a nullary Result.
   * @return True if Result is empty (a nullary Result) and not an actual
   *         result returned from a checkSat() (and friends) query.
   */
  bool isNull() const;

  /**
   * @return True if this result is from a satisfiable checkSat() or
   *         checkSatAssuming() query.
   */
  bool isSat() const;

  /**
   * @return True if this result is from an unsatisfiable checkSat() or
   *         checkSatAssuming() query.
   */
  bool isUnsat() const;

  /**
   * @return True if result is from a checkSat() or checkSatAssuming() query
   *         and cvc5 was not able to determine (un)satisfiability.
   */
  bool isUnknown() const;

  /**
   * Operator overloading for equality of two results.
   * @param r The result to compare to for equality.
   * @return True if the results are equal.
   */
  bool operator==(const Result& r) const;

  /**
   * Operator overloading for disequality of two results.
   * @param r The result to compare to for disequality.
   * @return True if the results are disequal.
   */
  bool operator!=(const Result& r) const;

  /**
   * @return An explanation for an unknown query result.
   */
  UnknownExplanation getUnknownExplanation() const;

  /**
   * @return A string representation of this result.
   */
  std::string toString() const;

 private:
  /**
   * Constructor.
   * @param r The internal result that is to be wrapped by this result.
   * @return The Result.
   */
  Result(const internal::Result& r);

  /**
   * The internal result wrapped by this result.
   *
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::Result`` is not ref counted.
   */
  std::shared_ptr d_result;
};

/**
 * Serialize a Result to given stream.
 * @param out The output stream.
 * @param r The result to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Result& r);

}  // namespace cvc5

namespace std {
/**
 * Hash function for results.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Result& result) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* SynthResult                                                                */
/* -------------------------------------------------------------------------- */

/**
 * Encapsulation of a solver synth result.
 *
 * This is the return value of the API functions:
 *   - Solver::checkSynth()
 *   - Solver::checkSynthNext()
 *
 * which we call "synthesis queries".  This class indicates whether the
 * synthesis query has a solution, has no solution, or is unknown.
 */
class CVC5_EXPORT SynthResult
{
  friend class Solver;

 public:
  /** Constructor. */
  SynthResult();

  /**
   * Determine if a given synthesis result is empty (a nullary result) and not
   * an actual result returned from a synthesis query.
   * @return True if SynthResult is null, i.e., not a SynthResult returned
   *         from a synthesis query.
   */
  bool isNull() const;

  /**
   * @return True if the synthesis query has a solution.
   */
  bool hasSolution() const;

  /**
   * @return True if the synthesis query has no solution. In this case, it
   *         was determined that there was no solution.
   */
  bool hasNoSolution() const;

  /**
   * @return True if the result of the synthesis query could not be determined.
   */
  bool isUnknown() const;

  /**
   * Operator overloading for equality of two synthesis results.
   * @param r The synthesis result to compare to for equality.
   * @return True if the synthesis results are equal.
   */
  bool operator==(const SynthResult& r) const;

  /**
   * Operator overloading for disequality of two synthesis results.
   * @param r The synthesis result to compare to for disequality.
   * @return True if the synthesis results are disequal.
   */
  bool operator!=(const SynthResult& r) const;

  /**
   * @return A string representation of this synthesis result.
   */
  std::string toString() const;

 private:
  /**
   * Constructor.
   * @param r The internal synth result that is to be wrapped by this synth.
   *          result
   * @return The SynthResult.
   */
  SynthResult(const internal::SynthResult& r);
  /**
   * The internal result wrapped by this result.
   *
   * @note This is a `std::shared_ptr` rather than a `std::unique_ptr`
   *       since `internal::SynthResult` is not ref counted.
   */
  std::shared_ptr d_result;
};

/**
 * Serialize a SynthResult to given stream.
 * @param out The output stream.
 * @param r The result to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const SynthResult& r);

}  // namespace cvc5

namespace std {
/**
 * Hash function for synthesis results.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::SynthResult& result) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Sort                                                                       */
/* -------------------------------------------------------------------------- */

class Datatype;

/**
 * The sort of a cvc5 term.
 */
class CVC5_EXPORT Sort
{
  friend class parser::Cmd;
  friend class DatatypeConstructor;
  friend class DatatypeConstructorDecl;
  friend class DatatypeSelector;
  friend class DatatypeDecl;
  friend class Datatype;
  friend class Op;
  friend class Term;
  friend class TermManager;
  friend class Solver;
  friend class Grammar;
  friend struct std::hash;

 public:
  /**
   * Constructor.
   */
  Sort();

  /**
   * Destructor.
   */
  ~Sort();

  /**
   * Comparison for structural equality.
   * @param s The sort to compare to.
   * @return True if the sorts are equal.
   */
  bool operator==(const Sort& s) const;

  /**
   * Comparison for structural disequality.
   * @param s The sort to compare to.
   * @return True if the sorts are not equal.
   */
  bool operator!=(const Sort& s) const;

  /**
   * Comparison for ordering on sorts.
   * @param s The sort to compare to.
   * @return True if this sort is less than s.
   */
  bool operator<(const Sort& s) const;

  /**
   * Comparison for ordering on sorts.
   * @param s The sort to compare to.
   * @return True if this sort is greater than s.
   */
  bool operator>(const Sort& s) const;

  /**
   * Comparison for ordering on sorts.
   * @param s The sort to compare to.
   * @return True if this sort is less than or equal to s.
   */
  bool operator<=(const Sort& s) const;

  /**
   * Comparison for ordering on sorts.
   * @param s The sort to compare to.
   * @return True if this sort is greater than or equal to s.
   */
  bool operator>=(const Sort& s) const;

  /**
   * Get the kind of this sort.
   * @return The kind of the sort.
   *
   * @warning This function is experimental and may change in future versions.
   */
  SortKind getKind() const;

  /**
   * Determine if this sort has a symbol (a name).
   *
   * For example, uninterpreted sorts and uninterpreted sort constructors have
   * symbols.
   *
   * @return True if the sort has a symbol.
   */
  bool hasSymbol() const;

  /**
   * Get the symbol of this Sort.
   *
   * @note Asserts hasSymbol().
   *
   * The symbol of this sort is the string that was
   * provided when constructing it via
   * Solver::mkUninterpretedSort(const std::string&) const,
   * Solver::mkUnresolvedSort(const std::string&, size_t) const, or
   * Solver::mkUninterpretedSortConstructorSort(const std::string&, size_t).
   *
   * @return The raw symbol of the sort.
   */
  std::string getSymbol() const;

  /**
   * Determine if this is the null sort (Sort::Sort()).
   * @return True if this Sort is the null sort.
   */
  bool isNull() const;

  /**
   * Determine if this is the Boolean sort (SMT-LIB: `Bool`).
   * @return True if this sort is the Boolean sort.
   */
  bool isBoolean() const;

  /**
   * Determine if this is the integer sort (SMT-LIB: `Int`).
   * @return True if this sort is the integer sort.
   */
  bool isInteger() const;

  /**
   * Determine if this is the real sort (SMT-LIB: `Real`).
   * @return True if this sort is the real sort.
   */
  bool isReal() const;

  /**
   * Determine if this is the string sort (SMT-LIB: `String`).
   * @return True if this sort is the string sort.
   */
  bool isString() const;

  /**
   * Determine if this is the regular expression sort (SMT-LIB: `RegLan`).
   * @return True if this sort is the regular expression sort.
   */
  bool isRegExp() const;

  /**
   * Determine if this is the rounding mode sort (SMT-LIB: `RoundingMode`).
   * @return True if this sort is the rounding mode sort.
   */
  bool isRoundingMode() const;

  /**
   * Determine if this is a bit-vector sort (SMT-LIB: `(_ BitVec i)`).
   * @return True if this sort is a bit-vector sort.
   */
  bool isBitVector() const;

  /**
   * Determine if this is a floatingpoint sort
   * (SMT-LIB: `(_ FloatingPoint eb sb)`).
   * @return True if this sort is a floating-point sort.
   */
  bool isFloatingPoint() const;

  /**
   * Determine if this is a datatype sort.
   * @return True if this sort is a datatype sort.
   */
  bool isDatatype() const;

  /**
   * Determine if this is a datatype constructor sort.
   * @return True if this sort is a datatype constructor sort.
   */
  bool isDatatypeConstructor() const;

  /**
   * Determine if this is a datatype selector sort.
   * @return True if this sort is a datatype selector sort.
   */
  bool isDatatypeSelector() const;

  /**
   * Determine if this is a datatype tester sort.
   * @return True if this sort is a datatype tester sort.
   */
  bool isDatatypeTester() const;
  /**
   * Determine if this is a datatype updater sort.
   * @return True if this sort is a datatype updater sort.
   */
  bool isDatatypeUpdater() const;
  /**
   * Determine if this is a function sort.
   * @return True if this sort is a function sort.
   */
  bool isFunction() const;

  /**
   * Determine if this is a predicate sort.
   *
   * A predicate sort is a function sort that maps to the Boolean sort. All
   * predicate sorts are also function sorts.
   *
   * @return True if this sort is a predicate sort.
   */
  bool isPredicate() const;

  /**
   * Determine if this is a tuple sort.
   * @return True if this sort is a tuple sort.
   */
  bool isTuple() const;

  /**
   * Determine if this is a nullable sort.
   * @return True if the sort is a nullable sort.
   */
  bool isNullable() const;

  /**
   * Determine if this is a record sort.
   * @warning This function is experimental and may change in future versions.
   * @return True if the sort is a record sort.
   */
  bool isRecord() const;

  /**
   * Determine if this is an array sort.
   * @return True if the sort is an array sort.
   */
  bool isArray() const;

  /**
   * Determine if this is a finite field sort.
   * @return True if the sort is a finite field sort.
   */
  bool isFiniteField() const;

  /**
   * Determine if this is a Set sort.
   * @return True if the sort is a Set sort.
   */
  bool isSet() const;

  /**
   * Determine if this is a Bag sort.
   * @return True if the sort is a Bag sort.
   */
  bool isBag() const;

  /**
   * Determine if this is a Sequence sort.
   * @return True if the sort is a Sequence sort.
   */
  bool isSequence() const;

  /**
   * Determine if this is an abstract sort.
   * @return True if the sort is a abstract sort.
   *
   * @warning This function is experimental and may change in future versions.
   */
  bool isAbstract() const;

  /**
   * Determine if this is an uninterpreted sort.
   * @return True if this is an uninterpreted sort.
   */
  bool isUninterpretedSort() const;

  /**
   * Determine if this is an uninterpreted sort constructor.
   *
   * An uninterpreted sort constructor has arity > 0 and can be instantiated to
   * construct uninterpreted sorts with given sort parameters.
   *
   * @return True if this is of sort constructor kind.
   */
  bool isUninterpretedSortConstructor() const;

  /**
   * Determine if this is an instantiated (parametric datatype or uninterpreted
   * sort constructor) sort.
   *
   * An instantiated sort is a sort that has been constructed from
   * instantiating a sort with sort arguments
   * (see Sort::instantiate(const std::vector&) const)).
   *
   * @return True if this is an instantiated sort.
   */
  bool isInstantiated() const;

  /**
   * Get the associated uninterpreted sort constructor of an instantiated
   * uninterpreted sort.
   *
   * @return The uninterpreted sort constructor sort.
   */
  Sort getUninterpretedSortConstructor() const;

  /**
   * @return The underlying datatype of a datatype sort.
   */
  Datatype getDatatype() const;

  /**
   * Instantiate a parameterized datatype sort or uninterpreted sort
   * constructor sort.
   *
   * Create sort parameters with Solver::mkParamSort().
   *
   * @param params The list of sort parameters to instantiate with.
   * @return The instantiated sort.
   */
  Sort instantiate(const std::vector& params) const;

  /**
   * Get the sorts used to instantiate the sort parameters of a parametric
   * sort (parametric datatype or uninterpreted sort constructor sort,
   * see Sort::instantiate(const std::vector& const)).
   *
   * @return The sorts used to instantiate the sort parameters of a
   *         parametric sort
   */
  std::vector getInstantiatedParameters() const;

  /**
   * Substitution of Sorts.
   *
   * Note that this replacement is applied during a pre-order traversal and
   * only once to the sort. It is not run until fix point.
   *
   * For example,
   * `(Array A B).substitute({A, C}, {(Array C D), (Array A B)})` will
   * return `(Array (Array C D) B)`.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param sort The subsort to be substituted within this sort.
   * @param replacement The sort replacing the substituted subsort.
   */
  Sort substitute(const Sort& sort, const Sort& replacement) const;

  /**
   * Simultaneous substitution of Sorts.
   *
   * Note that this replacement is applied during a pre-order traversal and
   * only once to the sort. It is not run until fix point. In the case that
   * sorts contains duplicates, the replacement earliest in the vector takes
   * priority.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param sorts The subsorts to be substituted within this sort.
   * @param replacements The sort replacing the substituted subsorts.
   */
  Sort substitute(const std::vector& sorts,
                  const std::vector& replacements) const;

  /**
   * Output a string representation of this sort to a given stream.
   * @param out The output stream.
   */
  void toStream(std::ostream& out) const;

  /**
   * @return A string representation of this sort.
   */
  std::string toString() const;

  /* Datatype constructor sort ------------------------------------------- */

  /**
   * @return The arity of a datatype constructor sort.
   */
  size_t getDatatypeConstructorArity() const;

  /**
   * @return The domain sorts of a datatype constructor sort.
   */
  std::vector getDatatypeConstructorDomainSorts() const;

  /**
   * @return The codomain sort of a constructor sort.
   */
  Sort getDatatypeConstructorCodomainSort() const;

  /* Selector sort ------------------------------------------------------- */

  /**
   * @return The domain sort of a datatype selector sort.
   */
  Sort getDatatypeSelectorDomainSort() const;

  /**
   * @return The codomain sort of a datatype selector sort.
   */
  Sort getDatatypeSelectorCodomainSort() const;

  /* Tester sort ------------------------------------------------------- */

  /**
   * @return The domain sort of a datatype tester sort.
   */
  Sort getDatatypeTesterDomainSort() const;

  /**
   * @return The codomain sort of a datatype tester sort, which is the Boolean
   *         sort.
   *
   * @note We mainly need this for the symbol table, which doesn't have
   *       access to the solver object.
   */
  Sort getDatatypeTesterCodomainSort() const;

  /* Function sort ------------------------------------------------------- */

  /**
   * @return The arity of a function sort.
   */
  size_t getFunctionArity() const;

  /**
   * @return The domain sorts of a function sort.
   */
  std::vector getFunctionDomainSorts() const;

  /**
   * @return The codomain sort of a function sort.
   */
  Sort getFunctionCodomainSort() const;

  /* Array sort ---------------------------------------------------------- */

  /**
   * @return The array index sort of an array sort.
   */
  Sort getArrayIndexSort() const;

  /**
   * @return The array element sort of an array sort.
   */
  Sort getArrayElementSort() const;

  /* Set sort ------------------------------------------------------------ */

  /**
   * @return The element sort of a set sort.
   */
  Sort getSetElementSort() const;

  /* Bag sort ------------------------------------------------------------ */

  /**
   * @return The element sort of a bag sort.
   */
  Sort getBagElementSort() const;

  /* Sequence sort ------------------------------------------------------- */

  /**
   * @return The element sort of a sequence sort.
   */
  Sort getSequenceElementSort() const;

  /* Abstract sort ------------------------------------------------------- */
  /**
   * @return The sort kind of an abstract sort, which denotes the kind of
   * sorts that this abstract sort denotes.
   *
   * @warning This function is experimental and may change in future versions.
   */
  SortKind getAbstractedKind() const;

  /* Uninterpreted sort constructor sort --------------------------------- */

  /**
   * @return The arity of an uninterpreted sort constructor sort.
   */
  size_t getUninterpretedSortConstructorArity() const;

  /* Bit-vector sort ----------------------------------------------------- */

  /**
   * @return The bit-width of the bit-vector sort.
   */
  uint32_t getBitVectorSize() const;

  /* Finite field sort --------------------------------------------------- */

  /**
   * @return The size of the finite field sort.
   */
  std::string getFiniteFieldSize() const;

  /* Floating-point sort ------------------------------------------------- */

  /**
   * @return The bit-width of the exponent of the floating-point sort.
   */
  uint32_t getFloatingPointExponentSize() const;

  /**
   * @return The width of the significand of the floating-point sort.
   */
  uint32_t getFloatingPointSignificandSize() const;

  /* Datatype sort ------------------------------------------------------- */

  /**
   * Get the arity of a datatype sort, which is the number of type parameters
   * if the datatype is parametric, or 0 otherwise.
   * @return The arity of a datatype sort.
   */
  size_t getDatatypeArity() const;

  /* Tuple sort ---------------------------------------------------------- */

  /**
   * @return The length of a tuple sort.
   */
  size_t getTupleLength() const;

  /**
   * @return The element sorts of a tuple sort.
   */
  std::vector getTupleSorts() const;

  /**
   * @return The element sort of a nullable sort.
   */
  Sort getNullableElementSort() const;

  /* --------------------------------------------------------------------- */

 private:
  /** @return The internal wrapped TypeNode of this sort. */
  const internal::TypeNode& getTypeNode(void) const;

  /** Helper to convert a vector of Sorts to internal TypeNodes. */
  std::vector static sortVectorToTypeNodes(
      const std::vector& sorts);
  /** Helper to convert a vector of internal TypeNodes to Sorts. */
  std::vector static typeNodeVectorToSorts(
      TermManager* tm, const std::vector& types);

  /**
   * Constructor.
   * @param tm The associated term manager.
   * @param t  The internal type that is to be wrapped by this sort.
   * @return The Sort.
   */
  Sort(TermManager* tm, const internal::TypeNode& t);

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal type wrapped by this sort.
   *
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr`` to
   *       avoid overhead due to memory allocation (``internal::Type`` is
   * already ref counted, so this could be a ``std::unique_ptr`` instead).
   */
  std::shared_ptr d_type;
};

/**
 * Serialize a sort to given stream.
 * @param out The output stream.
 * @param s The sort to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Sort& s);

}  // namespace cvc5

namespace std {

/**
 * Hash function for Sorts.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Sort& s) const;
};

}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Op                                                                     */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 operator.
 *
 * An operator is a term that represents certain operators, instantiated
 * with its required parameters, e.g., a Term of kind #BITVECTOR_EXTRACT.
 */
class CVC5_EXPORT Op
{
  friend class TermManager;
  friend class Term;
  friend struct std::hash;

 public:
  /**
   * Constructor.
   */
  Op();

  /**
   * Destructor.
   */
  ~Op();

  /**
   * Syntactic equality operator.
   *
   * @param t The operator to compare to for equality.
   * @return True if both operators are syntactically identical.
   */
  bool operator==(const Op& t) const;

  /**
   * Syntactic disequality operator.
   *
   * @param t The operator to compare to for disequality.
   * @return True if operators differ syntactically.
   */
  bool operator!=(const Op& t) const;

  /**
   * Get the kind of this operator.
   * @return The kind of this operator.
   */
  Kind getKind() const;

  /**
   * Determine if this operator is nullary.
   * @return True if this operator is a nullary operator.
   */
  bool isNull() const;

  /**
   * Determine if this operator is indexed.
   * @return True iff this operator is indexed.
   */
  bool isIndexed() const;

  /**
   * Get the number of indices of this operator.
   * @return The number of indices of this operator.
   */
  size_t getNumIndices() const;

  /**
   * Get the index at position `i` of an indexed operator.
   * @param i The position of the index to return.
   * @return The index at position i.
   */
  Term operator[](size_t i);

  /**
   * Get the string representation of this operator.
   * @return A string representation of this operator.
   */
  std::string toString() const;

 private:
  /**
   * Constructor for a single kind (non-indexed operator).
   * @param tm The associated term manager.
   * @param k  The kind of this Op.
   */
  Op(TermManager* tm, const Kind k);

  /**
   * Constructor.
   * @param tm The associated term managaer.
   * @param k The kind of this Op.
   * @param n The internal node that is to be wrapped by this term.
   * @return The Term.
   */
  Op(TermManager* tm, const Kind k, const internal::Node& n);

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * @note An indexed operator has a non-null internal node (``d_node``).
   *
   * @note We use a helper function to avoid having API functions call other API
   *       functions (we need to call this internally).
   *
   * @return True iff this Op is indexed.
   */
  bool isIndexedHelper() const;

  /**
   * Helper for getNumIndices
   * @return The number of indices of this op.
   */
  size_t getNumIndicesHelper() const;

  /**
   * Helper for operator[](size_t index).
   * @param index Position of the index. Should be less than
   *              getNumIndicesHelper().
   * @return The index at position index.
   */
  Term getIndexHelper(size_t index);

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /** The kind of this operator. */
  Kind d_kind;

  /**
   * The internal node wrapped by this operator.
   *
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr`` to
   *       avoid overhead due to memory allocation (``internal::Node`` is
   * already ref counted, so this could be a ``std::unique_ptr`` instead).
   */
  std::shared_ptr d_node;
};

/**
 * Serialize an operator to given stream.
 * @param out The output stream.
 * @param op  The operator to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Op& op);

}  // namespace cvc5

namespace std {
/**
 * Hash function for Ops.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Op& op) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Term                                                                       */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 Term.
 */
class CVC5_EXPORT Term
{
  friend class parser::Cmd;
  friend class Datatype;
  friend class DatatypeConstructor;
  friend class DatatypeSelector;
  friend class Proof;
  friend class TermManager;
  friend class Solver;
  friend class Grammar;
  friend class PluginInternal;
  friend class SynthResult;
  friend struct std::hash;

 public:
  /**
   * Constructor for a null term.
   */
  Term();

  /**
   * Destructor.
   */
  ~Term();

  /**
   * Syntactic equality operator.
   * @param t The term to compare to for equality.
   * @return True if the terms are equal.
   */
  bool operator==(const Term& t) const;

  /**
   * Syntactic disequality operator.
   * @param t The term to compare to for disequality.
   * @return True if terms are disequal.
   */
  bool operator!=(const Term& t) const;

  /**
   * Comparison for ordering on terms by their id.
   * @param t The term to compare to.
   * @return True if this term is less than t.
   */
  bool operator<(const Term& t) const;

  /**
   * Comparison for ordering on terms by their id.
   * @param t The term to compare to.
   * @return True if this term is greater than t.
   */
  bool operator>(const Term& t) const;

  /**
   * Comparison for ordering on terms by their id.
   * @param t The term to compare to.
   * @return True if this term is less than or equal to t.
   */
  bool operator<=(const Term& t) const;

  /**
   * Comparison for ordering on terms by their id.
   * @param t The term to compare to.
   * @return True if this term is greater than or equal to t.
   */
  bool operator>=(const Term& t) const;

  /**
   * Get the number of children of this term.
   * @return The number of children of this term.
   */
  size_t getNumChildren() const;

  /**
   * Get the child term of this term at a given index.
   * @param index The index of the child.
   * @return The child term at the given index.
   */
  Term operator[](size_t index) const;

  /**
   * Get the id of this term.
   * @return The id of this term.
   */
  uint64_t getId() const;

  /**
   * Get the kind of this term.
   * @return The kind of this term.
   */
  Kind getKind() const;

  /**
   * Get the sort of this term.
   * @return The sort of this term.
   */
  Sort getSort() const;

  /**
   * Replace `term` with `replacement` in this term.
   *
   * @param term        The term to replace.
   * @param replacement The term to replace it with.
   * @return The result of replacing `term` with `replacement` in this term.
   *
   * @note This replacement is applied during a pre-order traversal and
   *       only once (it is not run until fixed point).
   */
  Term substitute(const Term& term, const Term& replacement) const;

  /**
   * Simultaneously replace `terms` with `replacements` in this term.
   *
   * In the case that `terms` contains duplicates, the replacement earliest in
   * the vector takes priority. For example, calling substitute on `f(x,y)`
   * with `terms = { x, z }`, `replacements = { g(z), w }` results in the term
   * `f(g(z),y)`.
   *
   * @note Requires that `terms` and `replacements` are of equal size (they are
   *       interpreted as 1 : 1 mapping).
   *
   * @note This replacement is applied during a pre-order traversal and
   *       only once (it is not run until fixed point).
   *
   * @param terms        The terms to replace.
   * @param replacements The replacement terms.
   * @return The result of simultaneously replacing `terms` with `replacements`
   *         in this term.
   */
  Term substitute(const std::vector& terms,
                  const std::vector& replacements) const;

  /**
   * Determine if this term has an operator.
   * @return True iff this term has an operator.
   */
  bool hasOp() const;

  /**
   * Get the operator of a term with an operator.
   *
   * @note Requires that this term has an operator (see hasOp()).
   *
   * @return The Op used to create this term.
   */
  Op getOp() const;

  /**
   * Determine if this term has a symbol (a name).
   *
   * For example, free constants and variables have symbols.
   *
   * @return True if the term has a symbol.
   */
  bool hasSymbol() const;

  /**
   * Get the symbol of this Term.
   *
   * @note Requires that this term has a symbol (see hasSymbol()).
   *
   * The symbol of the term is the string that was provided when constructing
   * it via TermManager::mkConst() or TermManager::mkVar().
   *
   * @return The raw symbol of the term.
   */
  std::string getSymbol() const;

  /**
   * Determine if this term is nullary.
   * @return True if this Term is a null term.
   */
  bool isNull() const;

  /**
   * Boolean negation.
   * @return The Boolean negation of this term.
   */
  Term notTerm() const;

  /**
   * Boolean and.
   * @param t A Boolean term.
   * @return The conjunction of this term and the given term.
   */
  Term andTerm(const Term& t) const;

  /**
   * Boolean or.
   * @param t A Boolean term.
   * @return The disjunction of this term and the given term.
   */
  Term orTerm(const Term& t) const;

  /**
   * Boolean exclusive or.
   * @param t A Boolean term.
   * @return The exclusive disjunction of this term and the given term.
   */
  Term xorTerm(const Term& t) const;

  /**
   * Equality.
   * @param t A Boolean term.
   * @return A Boolean term representing equivalence of this term and the given
   *         term.
   */
  Term eqTerm(const Term& t) const;

  /**
   * Boolean implication.
   * @param t A Boolean term.
   * @return The implication of this term and the given term.
   */
  Term impTerm(const Term& t) const;

  /**
   * If-then-else with this term as the Boolean condition.
   * @param t The 'then' term.
   * @param e The 'else' term.
   * @return The if-then-else term with this term as the Boolean condition.
   */
  Term iteTerm(const Term& t, const Term& e) const;

  /**
   * @return A string representation of this term.
   */
  std::string toString() const;

  /**
   * Iterator for the children of a Term.
   * @note This treats uninterpreted functions as Term just like any other term
   *       for example, the term ``f(x, y)`` will have Kind ``APPLY_UF`` and
   *       three children: ``f``, ``x``, and ``y``
   */
  class CVC5_EXPORT const_iterator
  {
    friend class Term;

   public:
    /* The following types are required by trait std::iterator_traits */

    /** Iterator tag */
    using iterator_category = std::forward_iterator_tag;

    /** The type of the item */
    using value_type = Term;

    /** The pointer type of the item */
    using pointer = const Term*;

    /** The reference type of the item */
    using reference = const Term&;

    /** The type returned when two iterators are subtracted */
    using difference_type = std::ptrdiff_t;

    /* End of std::iterator_traits required types */

    /**
     * Null Constructor.
     */
    const_iterator();

    /**
     * Constructor
     * @param tm The associated term manager.
     * @param e  A `std::shared pointer` to the node that we're iterating over.
     * @param p  The position of the iterator (e.g. which child it's on).
     */
    const_iterator(TermManager* tm,
                   const std::shared_ptr& e,
                   uint32_t p);

    /**
     * Copy constructor.
     */
    const_iterator(const const_iterator& it);

    /**
     * Assignment operator.
     * @param it The iterator to assign to.
     * @return The reference to the iterator after assignment.
     */
    const_iterator& operator=(const const_iterator& it);

    /**
     * Equality operator.
     * @param it The iterator to compare to for equality.
     * @return True if the iterators are equal.
     */
    bool operator==(const const_iterator& it) const;

    /**
     * Disequality operator.
     * @param it The iterator to compare to for disequality.
     * @return True if the iterators are disequal.
     */
    bool operator!=(const const_iterator& it) const;

    /**
     * Increment operator (prefix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator& operator++();

    /**
     * Increment operator (postfix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator operator++(int);

    /**
     * Dereference operator.
     * @return The term this iterator points to.
     */
    Term operator*() const;

   private:
    /**
     * The associated term manager.
     */
    TermManager* d_tm = nullptr;
    /** The original node to be iterated over. */
    std::shared_ptr d_origNode;
    /** Keeps track of the iteration position. */
    uint32_t d_pos;
  };

  /**
   * @return An iterator to the first child of this Term.
   */
  const_iterator begin() const;

  /**
   * @return An iterator to one-off-the-last child of this Term.
   */
  const_iterator end() const;

  /**
   * Get the sign of an integer or real value.
   * @note Requires that this term is an integer or real value.
   * @return 0 if this term is zero, -1 if this term is a negative real or
   *         integer value, 1 if this term is a positive real or integer value.
   */
  int32_t getRealOrIntegerValueSign() const;
  /**
   * Determine if this term is an int32 value.
   * @note This will return true for integer constants and real constants that
   *       have integer value.
   * @return True if the term is an integral value that fits within int32_t.
   */
  bool isInt32Value() const;
  /**
   * Get the `int32_t` representation of this integral value.
   * @note Requires that this term is an int32 value (see isInt32Value()).
   * @return This integral value as `int32_t` value.
   */
  int32_t getInt32Value() const;
  /**
   * Determine if this term is a uint32 value.
   * @note This will return true for integer constants and real constants that
   *       have integral value.
   * @return True if the term is an integral value and fits within uint32_t.
   */
  bool isUInt32Value() const;
  /**
   * Get the `uint32_t` representation of this integral value.
   * @note Requires that this term is a uint32 value (see isUInt32Value()).
   * @return This integral value as a `uint32_t`.
   */
  uint32_t getUInt32Value() const;
  /**
   * Determine if this term is an int64 value.
   * @note This will return true for integer constants and real constants that
   *       have integral value.
   * @return True if the term is an integral value and fits within int64_t.
   */
  bool isInt64Value() const;
  /**
   * Get the `int64_t` representation of this integral value.
   * @note Requires that this term is an int64 value (see isInt64Value()).
   * @return This integral value as a `int64_t`.
   */
  int64_t getInt64Value() const;
  /**
   * Determine if this term is a uint64 value.
   * @note This will return true for integer constants and real constants that
   *       have integral value.
   * @return True if the term is an integral value that fits within uint64_t.
   */
  bool isUInt64Value() const;
  /**
   * Get the `uint64_t` representation of this integral value.
   * @note Requires that this term is an uint64 value (see isUInt64Value()).
   * @return This integral value as a `uint64_t`.
   */
  uint64_t getUInt64Value() const;
  /**
   * Determine if this term is an integral value.
   * @return True if the term is an integer constant or a real constant that
   * has an integral value.
   */
  bool isIntegerValue() const;
  /**
   * Get a string representation of this integral value.
   * @note Requires that this term is an integral value (see isIntegerValue()).
   * @return The integral term in (decimal) string representation.
   */
  std::string getIntegerValue() const;

  /**
   * Determine if this term is a string value.
   * @return True if the term is a string value.
   */
  bool isStringValue() const;
  /**
   * Get the native string representation of a string value.
   * @note Requires that this term is a string value (see isStringValue()).
   * @note This is not to be confused with toString(), which returns
   *       some string representation of the term, whatever data it may hold.
   * @return The string term as a native string value.
   */
  std::wstring getStringValue() const;

  /**
   * Determine if this term is a rational value whose numerator fits into an
   * int32 value and its denominator fits into a uint32 value.
   * @return True if the term is a rational and its numerator and denominator
   *         fit into 32 bit integer values.
   */
  bool isReal32Value() const;
  /**
   * Get the 32 bit integer representations of the numerator and denominator of
   * a rational value.
   * @note Requires that this term is a rational value and its numerator and
   *       denominator fit into 32 bit integer values (see isReal32Value()).
   * @return The representation of a rational value as a pair of its numerator
   *         and denominator as integer values.
   */
  std::pair getReal32Value() const;
  /**
   * Determine if this term is a rational value whose numerator fits into an
   * int64 value and its denominator fits into a uint64 value.
   * @return True if the term is a rational value whose numerator and
   *         denominator fit within int64_t and uint64_t, respectively.
   */
  bool isReal64Value() const;
  /**
   * Get the 64 bit integer representations of the numerator and denominator of
   * a rational value.
   * @note Requires that this term is a rational value and its numerator and
   *       denominator fit into 64 bit integer values (see isReal64Value()).
   * @return The representation of a rational value as a pair of its numerator
   *         and denominator.
   */
  std::pair getReal64Value() const;
  /**
   * Determine if this term is a rational value.
   * @note A term of kind PI is not considered to be a real value.
   * @return True if the term is a rational value.
   */
  bool isRealValue() const;
  /**
   * Get a string representation of this rational value.
   * @note Requires that this term is a rational value (see isRealValue()).
   * @return The representation of a rational value as a (rational) string.
   */
  std::string getRealValue() const;

  /**
   * Determine if this term is a constant array.
   * @return True if the term is a constant array.
   */
  bool isConstArray() const;
  /**
   * Determine the base (element stored at all indices) of a constant array.
   * @note Requires that this term is a constant array (see isConstArray()).
   * @return The base term.
   */
  Term getConstArrayBase() const;

  /**
   * Determine if this term is a Boolean value.
   * @return True if the term is a Boolean value.
   */
  bool isBooleanValue() const;
  /**
   * Get the value of a Boolean term as a native Boolean value.
   * @note Asserts isBooleanValue().
   * @return The representation of a Boolean value as a native Boolean value.
   */
  bool getBooleanValue() const;

  /**
   * Determine if this term is a bit-vector value.
   * @return True if the term is a bit-vector value.
   */
  bool isBitVectorValue() const;
  /**
   * Get the string representation of a bit-vector value.
   *
   * @note Asserts isBitVectorValue().
   * @param base `2` for binary, `10` for decimal, and `16` for hexadecimal.
   * @return The string representation of a bit-vector value.
   */
  std::string getBitVectorValue(uint32_t base = 2) const;

  /**
   * Determine if this term is a finite field value.
   * @return True if the term is a finite field value.
   */
  bool isFiniteFieldValue() const;
  /**
   * Get the string representation of a finite field value (base 10).
   *
   * @note Asserts isFiniteFieldValue().
   *
   * @note Uses the integer representative of smallest absolute value.
   *
   * @return The string representation of the integer representation of this
   * finite field value.
   */
  std::string getFiniteFieldValue() const;

  /**
   * Determine if this term is an uninterpreted sort value.
   * @return True if the term is an abstract value.
   */
  bool isUninterpretedSortValue() const;
  /**
   * Get a string representation of an uninterpreted sort value.
   * @note Asserts isUninterpretedSortValue().
   * @return The representation of an uninterpreted sort value as a string.
   */
  std::string getUninterpretedSortValue() const;

  /**
   * Determine if this term is a tuple value.
   * @return True if the term is a tuple value.
   */
  bool isTupleValue() const;
  /**
   * Get a tuple value as a vector of terms.
   * @note Asserts isTupleValue().
   * @return The representation of a tuple value as a vector of terms.
   */
  std::vector getTupleValue() const;

  /**
   * Determine if this term is a floating-point rounding mode value.
   * @return True if the term is a rounding mode value.
   */
  bool isRoundingModeValue() const;
  /**
   * Get the RoundingMode value of a given rounding-mode value term.
   * @note Asserts isRoundingModeValue().
   * @return The floating-point rounding mode value of the term.
   */
  RoundingMode getRoundingModeValue() const;

  /**
   * Determine if this term is a floating-point positive zero value (+zero).
   * @return True if the term is the floating-point value for positive zero.
   */
  bool isFloatingPointPosZero() const;
  /**
   * Determine if this term is a floating-point negative zero value (-zero).
   * @return True if the term is the floating-point value for negative zero.
   */
  bool isFloatingPointNegZero() const;
  /**
   * Determine if this term is a floating-point positive infinity value (+oo).
   * @return True if the term is the floating-point value for positive.
   * infinity.
   */
  bool isFloatingPointPosInf() const;
  /**
   * Determine if this term is a floating-point negative infinity value (-oo).
   * @return True if the term is the floating-point value for negative.
   * infinity.
   */
  bool isFloatingPointNegInf() const;
  /**
   * Determine if a given term is a floating-point NaN value.
   * @return True if the term is the floating-point value for not a number.
   */
  bool isFloatingPointNaN() const;
  /**
   * Determine if a given term is a floating-point value.
   * @return True if the term is a floating-point value.
   */
  bool isFloatingPointValue() const;
  /**
   * Get the representation of a floating-point value as a tuple of its
   * exponent width, significand width and a bit-vector value term.
   * @note Asserts isFloatingPointValue().
   * @return The floating-point value representation.
   */
  std::tuple getFloatingPointValue() const;

  /**
   * Determine if this term is a set value.
   *
   * A term is a set value if it is considered to be a (canonical) constant set
   * value.  A canonical set value is one whose AST is:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (union (singleton c1) ... (union (singleton c_{n-1}) (singleton c_n))))
   * \endverbatim
   *
   * where @f$c_1 ... c_n@f$ are values ordered by id such that
   * @f$c_1 > ... > c_n@f$ (see @ref Term::operator>(const Term&) const).
   *
   * @note A universe set term (kind `SET_UNIVERSE`) is not considered to be
   *       a set value.
   *
   * @return True if the term is a set value.
   */
  bool isSetValue() const;
  /**
   * Get a set value as a set of terms.
   * @note Asserts isSetValue().
   * @return The representation of a set value as a set of terms.
   */
  std::set getSetValue() const;

  /**
   * Determine if this term is a sequence value.
   *
   * A term is a sequence value if it has kind #CONST_SEQUENCE. In contrast to
   * values for the set sort (as described in isSetValue()), a sequence value
   * is represented as a Term with no children.
   *
   * Semantically, a sequence value is a concatenation of unit sequences
   * whose elements are themselves values. For example:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (seq.++ (seq.unit 0) (seq.unit 1))
   * \endverbatim
   *
   * The above term has two representations in Term. One is as the sequence
   * concatenation term:
   *
   * \rst
   * .. code:: lisp
   *
   *     (SEQ_CONCAT (SEQ_UNIT 0) (SEQ_UNIT 1))
   * \endrst
   *
   * where 0 and 1 are the terms corresponding to the integer constants 0 and 1.
   *
   * Alternatively, the above term is represented as the constant sequence
   * value:
   *
   * \rst
   * .. code:: lisp
   *
   *     CONST_SEQUENCE_{0,1}
   * \endrst
   *
   * where calling getSequenceValue() on the latter returns the vector `{0, 1}`.
   *
   * The former term is not a sequence value, but the latter term is.
   *
   * Constant sequences cannot be constructed directly via the API. They are
   * returned in response to API calls such Solver::getValue() and
   * Solver::simplify().
   *
   * @return True if the term is a sequence value.
   */
  bool isSequenceValue() const;
  /**
   * Get a sequence value as a vector of terms.
   * @note Asserts isSequenceValue().
   * @return The representation of a sequence value as a vector of terms.
   */
  std::vector getSequenceValue() const;

  /**
   * Determine if this term is a cardinality constraint.
   * @return True if the term is a cardinality constraint.
   */
  bool isCardinalityConstraint() const;
  /**
   * Get a cardinality constraint as a pair of its sort and upper bound.
   * @note Asserts isCardinalityConstraint().
   * @return The sort the cardinality constraint is for and its upper bound.
   */
  std::pair getCardinalityConstraint() const;

  /**
   * Determine if this term is a real algebraic number.
   * @return True if the term is a real algebraic number.
   */
  bool isRealAlgebraicNumber() const;
  /**
   * Get the defining polynomial for a real algebraic number term, expressed in
   * terms of the given variable.
   * @note Asserts isRealAlgebraicNumber().
   * @param v The variable over which to express the polynomial.
   * @return The defining polynomial.
   */
  Term getRealAlgebraicNumberDefiningPolynomial(const Term& v) const;
  /**
   * Get the lower bound for a real algebraic number value.
   * @note Asserts isRealAlgebraicNumber().
   * @return The lower bound.
   */
  Term getRealAlgebraicNumberLowerBound() const;
  /**
   * Get the upper bound for a real algebraic number value.
   * @note Asserts isRealAlgebraicNumber().
   * @return The upper bound.
   */
  Term getRealAlgebraicNumberUpperBound() const;

  /**
   * Is this term a skolem?
   * @warning This function is experimental and may change in future versions.
   * @return True if this term is a skolem function.
   */
  bool isSkolem() const;
  /**
   * Get skolem identifier of this term.
   * @note Asserts isSkolem().
   * @warning This function is experimental and may change in future versions.
   * @return The skolem identifier of this term.
   */
  SkolemId getSkolemId() const;
  /**
   * Get the skolem indices of this term.
   * @note Asserts isSkolem().
   * @warning This function is experimental and may change in future versions.
   * @return The skolem indices of this term. This is list of terms that the
   * skolem function is indexed by. For example, the array diff skolem
   * `SkolemId::ARRAY_DEQ_DIFF` is indexed by two arrays.
   */
  std::vector getSkolemIndices() const;

 protected:
  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

 private:
  /** Helper function to collect all elements of a set. */
  static void collectSet(std::set& set,
                         const internal::Node& node,
                         TermManager* tm);
  /** Helper function to collect all elements of a sequence. */
  static void collectSequence(std::vector& seq,
                              const internal::Node& node,
                              TermManager* tm);

  /**
   * Constructor.
   * @param tm The associated term manager.
   * @param n The internal node that is to be wrapped by this term.
   * @return The Term.
   */
  Term(TermManager* tm, const internal::Node& n);

  /** @return The internal wrapped Node of this term. */
  const internal::Node& getNode(void) const;

  /** Helper to convert a vector of Terms to internal Nodes. */
  std::vector static termVectorToNodes(
      const std::vector& terms);
  /** Helper to convert a vector of internal Nodes to Terms. */
  std::vector static nodeVectorToTerms(
      TermManager* tm, const std::vector& nodes);

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * Helper function that returns the kind of the term, which takes into
   * account special cases of the conversion for internal to external kinds.
   * @return The kind of this term.
   */
  Kind getKindHelper() const;

  /**
   * The internal node wrapped by this term.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr`` to
   *       avoid overhead due to memory allocation (``internal::Node`` is
   * already ref counted, so this could be a ``std::unique_ptr`` instead).
   */
  std::shared_ptr d_node;
};

/**
 * Serialize a term to given stream.
 * @param out The output stream.
 * @param t The term to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Term& t);

/**
 * Serialize a vector of terms to given stream.
 * @param out The output stream.
 * @param vector The vector of terms to be serialized to the given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out, const std::vector& vector);

/**
 * Serialize a set of terms to the given stream.
 * @param out The output stream.
 * @param set The set of terms to be serialized to the given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out, const std::set& set);

/**
 * Serialize an unordered_set of terms to the given stream.
 *
 * @param out The output stream.
 * @param unordered_set The set of terms to be serialized to the given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out,
                         const std::unordered_set& unordered_set);

/**
 * Serialize a map of terms to the given stream.
 *
 * @param out The output stream.
 * @param map The map of terms to be serialized to the given stream.
 * @return The output stream.
 */
template 
CVC5_EXPORT std::ostream& operator<<(std::ostream& out,
                                     const std::map& map);

/**
 * Serialize an unordered_map of terms to the given stream.
 *
 * @param out The output stream.
 * @param unordered_map The map of terms to be serialized to the given stream.
 * @return The output stream.
 */
template 
CVC5_EXPORT std::ostream& operator<<(
    std::ostream& out, const std::unordered_map& unordered_map);

}  // namespace cvc5

namespace std {
/**
 * Hash function for Terms.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Term& t) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Datatypes                                                                  */
/* -------------------------------------------------------------------------- */

class DatatypeConstructorIterator;
class DatatypeIterator;

/**
 * A cvc5 datatype constructor declaration. A datatype constructor declaration
 * is a specification used for creating a datatype constructor.
 */
class CVC5_EXPORT DatatypeConstructorDecl
{
  friend class DatatypeDecl;
  friend class TermManager;
  friend class Solver;
  friend struct std::hash;

 public:
  /** Constructor.  */
  DatatypeConstructorDecl();

  /**
   * Destructor.
   */
  ~DatatypeConstructorDecl();

  /**
   * Equality operator.
   * @param decl The datatype constructor declaration to compare to for
   *             equality.
   * @return True if the datatype constructor declarations are equal.
   */
  bool operator==(const DatatypeConstructorDecl& decl) const;

  /**
   * Add datatype selector declaration.
   * @param name The name of the datatype selector declaration to add.
   * @param sort The codomain sort of the datatype selector declaration to add.
   */
  void addSelector(const std::string& name, const Sort& sort);
  /**
   * Add datatype selector declaration whose codomain type is the datatype
   * itself.
   * @param name The name of the datatype selector declaration to add.
   */
  void addSelectorSelf(const std::string& name);

  /**
   * Add datatype selector declaration whose codomain sort is an unresolved
   * datatype with the given name.
   * @param name The name of the datatype selector declaration to add.
   * @param unresDataypeName The name of the unresolved datatype. The codomain
   *                         of the selector will be the resolved datatype with
   *                         the given name.
   */
  void addSelectorUnresolved(const std::string& name,
                             const std::string& unresDataypeName);

  /**
   * @return True if this DatatypeConstructorDecl is a null declaration.
   */
  bool isNull() const;

  /**
   * @return A string representation of this datatype constructor declaration.
   */
  std::string toString() const;

 private:
  /**
   * Constructor.
   * @param tm   The associated term manager.
   * @param name The name of the datatype constructor.
   * @return The DatatypeConstructorDecl.
   */
  DatatypeConstructorDecl(TermManager* tm, const std::string& name);

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * Is the underlying constructor resolved (i.e,. has it been used to declare
   * a datatype already)?
   */
  bool isResolved() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal (intermediate) datatype constructor wrapped by this
   * datatype constructor declaration.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::DTypeConstructor`` is not ref counted.
   */
  std::shared_ptr d_ctor;
};

}  // namespace cvc5

namespace std {
/**
 * Hash function for datatype constructor declarations.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::DatatypeConstructorDecl& decl) const;
};
}  // namespace std

namespace cvc5 {

class Solver;

/**
 * A cvc5 datatype declaration. A datatype declaration is not itself a datatype
 * (see Datatype), but a specification for creating a datatype sort.
 *
 * The interface for a datatype declaration coincides with the syntax for the
 * SMT-LIB 2.6 command `declare-datatype`, or a single datatype within the
 * `declare-datatypes` command.
 *
 * Datatype sorts can be constructed from a DatatypeDecl using:
 *   - Solver::mkDatatypeSort()
 *   - Solver::mkDatatypeSorts()
 */
class CVC5_EXPORT DatatypeDecl
{
  friend class DatatypeConstructorArg;
  friend class TermManager;
  friend class Solver;
  friend struct std::hash;

 public:
  /** Constructor.  */
  DatatypeDecl();

  /**
   * Destructor.
   */
  ~DatatypeDecl();

  /**
   * Equality operator.
   * @param decl The datatype declaration to compare to for equality.
   * @return True if the datatype declarations are equal.
   */
  bool operator==(const DatatypeDecl& decl) const;

  /**
   * Add datatype constructor declaration.
   * @param ctor The datatype constructor declaration to add.
   */
  void addConstructor(const DatatypeConstructorDecl& ctor);

  /**
   * Get the number of constructors (so far) for this Datatype declaration.
   * @return The number of constructors.
   */
  size_t getNumConstructors() const;

  /**
   * Determine if this Datatype declaration is parametric.
   * @warning This function is experimental and may change in future versions.
   * @return True if this datatype declaration is parametric.
   */
  bool isParametric() const;

  /**
   * Determine if this datatype declaration is resolved (has already been used
   * to declare a datatype).
   * @return True if this datatype declaration is resolved.
   */
  bool isResolved() const;

  /**
   * Determine if this datatype declaration is nullary.
   * @return True if this DatatypeDecl is a null object.
   */
  bool isNull() const;

  /**
   * Get a string representation of this datatype declaration.
   * @return A string representation.
   */
  std::string toString() const;

  /**
   * Get the name of this datatype declaration.
   * @return The name.
   */
  std::string getName() const;

 private:
  /**
   * Constructor.
   * @param tm   The associated term manager.
   * @param name The name of the datatype.
   * @param isCoDatatype True if a codatatype is to be constructed.
   * @return The DatatypeDecl.
   */
  DatatypeDecl(TermManager* tm,
               const std::string& name,
               bool isCoDatatype = false);

  /**
   * Constructor for parameterized datatype declaration.
   * Create sorts parameter with Solver::mkParamSort().
   * @param tm   The associated term manager.
   * @param name The name of the datatype.
   * @param params A list of sort parameters.
   * @param isCoDatatype True if a codatatype is to be constructed.
   */
  DatatypeDecl(TermManager* tm,
               const std::string& name,
               const std::vector& params,
               bool isCoDatatype = false);

  /** @return The internal wrapped Dtype of this datatype declaration. */
  internal::DType& getDatatype(void) const;

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal (intermediate) datatype wrapped by this datatype
   * declaration.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::DType`` is not ref counted.
   */
  std::shared_ptr d_dtype;
};

}  // namespace cvc5

namespace std {
/**
 * Hash function for datatype declarations.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::DatatypeDecl& decl) const;
};
}  // namespace std

namespace cvc5 {

/**
 * A cvc5 datatype selector.
 */
class CVC5_EXPORT DatatypeSelector
{
  friend class Datatype;
  friend class DatatypeConstructor;
  friend class TermManager;
  friend struct std::hash;

 public:
  /**
   * Constructor.
   */
  DatatypeSelector();

  /**
   * Destructor.
   */
  ~DatatypeSelector();

  /**
   * Equality operator.
   * @param sel The datatype selector to compare to for equality.
   * @return True if the datatype selectors are equal.
   */
  bool operator==(const DatatypeSelector& sel) const;

  /**
   * Get the name of this datatype selector.
   * @return The name of this Datatype selector.
   */
  std::string getName() const;

  /**
   * Get the selector term of this datatype selector.
   *
   * Selector terms are a class of function-like terms of selector
   * sort (Sort::isDatatypeSelector()), and should be used as the first
   * argument of Terms of kind #APPLY_SELECTOR.
   *
   * @return The selector term.
   */
  Term getTerm() const;

  /**
   * Get the updater term of this datatype selector.
   *
   * Similar to selectors, updater terms are a class of function-like terms of
   * updater Sort (Sort::isDatatypeUpdater()), and should be used as the first
   * argument of Terms of kind #APPLY_UPDATER.
   *
   * @return The updater term.
   */
  Term getUpdaterTerm() const;

  /**
   * Get the codomain sort of this selector.
   * @return The codomain sort of this selector.
   */
  Sort getCodomainSort() const;

  /**
   * @return True if this DatatypeSelector is a null object.
   */
  bool isNull() const;

  /**
   * Get the string representation of this datatype selector.
   * @return The string representation.
   */
  std::string toString() const;

 private:
  /**
   * Constructor.
   * @param tm   The associated term manager.
   * @param stor The internal datatype selector to be wrapped.
   * @return The DatatypeSelector.
   */
  DatatypeSelector(TermManager* tm, const internal::DTypeSelector& stor);

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal datatype selector wrapped by this datatype selector.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::DType`` is not ref counted.
   */
  std::shared_ptr d_stor;
};

}  // namespace cvc5

namespace std {
/**
 * Hash function for datatype Selectors.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::DatatypeSelector& sel) const;
};
}  // namespace std

namespace cvc5 {

/**
 * A cvc5 datatype constructor.
 */
class CVC5_EXPORT DatatypeConstructor
{
  friend class Datatype;
  friend class TermManager;
  friend struct std::hash;

 public:
  /**
   * Constructor.
   */
  DatatypeConstructor();

  /**
   * Destructor.
   */
  ~DatatypeConstructor();

  /**
   * Equality operator.
   * @param cons The datatype constructor to compare to for equality.
   * @return True if the datatype constructors are equal.
   */
  bool operator==(const DatatypeConstructor& cons) const;

  /**
   * Get the name of this datatype constructor.
   * @return The name.
   */
  std::string getName() const;

  /**
   * Get the constructor term of this datatype constructor.
   *
   * Datatype constructors are a special class of function-like terms whose sort
   * is datatype constructor (Sort::isDatatypeConstructor()). All datatype
   * constructors, including nullary ones, should be used as the
   * first argument to Terms whose kind is #APPLY_CONSTRUCTOR. For example,
   * the nil list can be constructed by
   * `Solver::mkTerm(Kind::APPLY_CONSTRUCTOR, {t})`, where `t` is the term
   * returned by this function.
   *
   * @note This function should not be used for parametric datatypes. Instead,
   *       use the function DatatypeConstructor::getInstantiatedTerm() below.
   *
   * @return The constructor term.
   */
  Term getTerm() const;

  /**
   * Get the constructor term of this datatype constructor whose return
   * type is `retSort`.
   *
   * This function is intended to be used on constructors of parametric
   * datatypes and can be seen as returning the constructor term that has been
   * explicitly cast to the given sort.
   *
   * This function is required for constructors of parametric datatypes whose
   * return type cannot be determined by type inference. For example, given:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-datatype List
   *         (par (T) ((nil) (cons (head T) (tail (List T))))))
   * \endverbatim
   *
   * The type of nil terms must be provided by the user. In SMT version 2.6,
   * this is done via the syntax for qualified identifiers:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (as nil (List Int))
   * \endverbatim
   *
   * This function is equivalent of applying the above, where this
   * DatatypeConstructor is the one corresponding to `nil`, and `retSort` is
   * `(List Int)`.
   *
   * @note The returned constructor term `t` is used to construct the above
   *       (nullary) application of `nil` with
   *       `Solver::mkTerm(Kind::APPLY_CONSTRUCTOR, {t})`.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param retSort The desired return sort of the constructor.
   * @return The constructor term.
   */
  Term getInstantiatedTerm(const Sort& retSort) const;

  /**
   * Get the tester term of this datatype constructor.
   *
   * Similar to constructors, testers are a class of function-like terms of
   * tester sort (Sort::isDatatypeConstructor()) which should be used as the
   * first argument of Terms of kind #APPLY_TESTER.
   *
   * @return The tester term.
   */
  Term getTesterTerm() const;

  /**
   * @return The number of selectors (so far) of this Datatype constructor.
   */
  size_t getNumSelectors() const;

  /** @return The i^th DatatypeSelector. */
  DatatypeSelector operator[](size_t index) const;
  /**
   * Get the datatype selector with the given name.
   * @note This is a linear search through the selectors, so in case of
   *       multiple, similarly-named selectors, the first is returned.
   * @param name The name of the datatype selector.
   * @return The first datatype selector with the given name.
   */
  DatatypeSelector operator[](const std::string& name) const;
  /**
   * Get the datatype selector with the given name.
   * @note This is a linear search through the selectors, so in case of
   *       multiple, similarly-named selectors, the first is returned.
   * @param name The name of the datatype selector.
   * @return The first datatype selector with the given name.
   */
  DatatypeSelector getSelector(const std::string& name) const;

  /**
   * @return True if this DatatypeConstructor is a null object.
   */
  bool isNull() const;

  /**
   * Get a string representation of this datatype constructor.
   * @return The string representation.
   */
  std::string toString() const;

  /**
   * Iterator for the selectors of a datatype constructor.
   */
  class CVC5_EXPORT const_iterator
  {
    friend class DatatypeConstructor;  // to access constructor

   public:
    /* The following types are required by trait std::iterator_traits */

    /** Iterator tag */
    using iterator_category = std::forward_iterator_tag;

    /** The type of the item */
    using value_type = DatatypeConstructor;

    /** The pointer type of the item */
    using pointer = const DatatypeConstructor*;

    /** The reference type of the item */
    using reference = const DatatypeConstructor&;

    /** The type returned when two iterators are subtracted */
    using difference_type = std::ptrdiff_t;

    /* End of std::iterator_traits required types */

    /** Nullary constructor (required for Cython). */
    const_iterator();

    /**
     * Assignment operator.
     * @param it The iterator to assign to.
     * @return The reference to the iterator after assignment.
     */
    const_iterator& operator=(const const_iterator& it);

    /**
     * Equality operator.
     * @param it The iterator to compare to for equality.
     * @return True if the iterators are equal.
     */
    bool operator==(const const_iterator& it) const;

    /**
     * Disequality operator.
     * @param it The iterator to compare to for disequality.
     * @return True if the iterators are disequal.
     */
    bool operator!=(const const_iterator& it) const;

    /**
     * Increment operator (prefix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator& operator++();

    /**
     * Increment operator (postfix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator operator++(int);

    /**
     * Dereference operator.
     * @return A reference to the selector this iterator points to.
     */
    const DatatypeSelector& operator*() const;

    /**
     * Dereference operator.
     * @return A pointer to the selector this iterator points to.
     */
    const DatatypeSelector* operator->() const;

   private:
    /**
     * Constructor.
     * @param tm   The associated term manager.
     * @param ctor The internal datatype constructor to iterate over.
     * @param begin True if this is a `begin()` iterator.
     */
    const_iterator(TermManager* tm,
                   const internal::DTypeConstructor& ctor,
                   bool begin);

    /**
     * The associated term manager.
     */
    TermManager* d_tm = nullptr;

    /**
     * A pointer to the list of selectors of the internal datatype
     * constructor to iterate over.
     * This pointer is maintained for operators == and != only.
     */
    const void* d_int_stors;

    /** The list of datatype selector (wrappers) to iterate over. */
    std::vector d_stors;

    /** The current index of the iterator. */
    size_t d_idx;
  };

  /**
   * @return An iterator to the first selector of this constructor.
   */
  const_iterator begin() const;

  /**
   * @return An iterator to one-off-the-last selector of this constructor.
   */
  const_iterator end() const;

 private:
  /**
   * Constructor.
   * @param tm   The associated term manager.
   * @param ctor The internal datatype constructor to be wrapped.
   * @return The DatatypeConstructor.
   */
  DatatypeConstructor(TermManager* tm, const internal::DTypeConstructor& ctor);

  /**
   * Return selector for name.
   * @param name The name of selector to find.
   * @return The selector object for the name.
   */
  DatatypeSelector getSelectorForName(const std::string& name) const;

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal datatype constructor wrapped by this datatype constructor.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::DType`` is not ref counted.
   */
  std::shared_ptr d_ctor;
};

}  // namespace cvc5

namespace std {
/**
 * Hash function for datatype constructors.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::DatatypeConstructor& cons) const;
};
}  // namespace std

namespace cvc5 {

/**
 * A cvc5 datatype.
 */
class CVC5_EXPORT Datatype
{
  friend class TermManager;
  friend class Sort;
  friend struct std::hash;

 public:
  /** Constructor. */
  Datatype();

  /**
   * Destructor.
   */
  ~Datatype();

  /**
   * Equality operator.
   * @param dt The datatype to compare to for equality.
   * @return True if the datatypes are equal.
   */
  bool operator==(const Datatype& dt) const;

  /**
   * Get the datatype constructor at a given index.
   * @param idx The index of the datatype constructor to return.
   * @return The datatype constructor with the given index.
   */
  DatatypeConstructor operator[](size_t idx) const;

  /**
   * Get the datatype constructor with the given name.
   * @note This is a linear search through the constructors, so in case of
   *       multiple, similarly-named constructors, the first is returned.
   * @param name The name of the datatype constructor.
   * @return The datatype constructor with the given name.
   */
  DatatypeConstructor operator[](const std::string& name) const;
  DatatypeConstructor getConstructor(const std::string& name) const;

  /**
   * Get the datatype selector with the given name.
   * @note This is a linear search through the constructors and their
   *       selectors, so in case of multiple, similarly-named selectors, the
   *       first is returned.
   * @param name The name of the datatype selector.
   * @return The datatype selector with the given name.
   */
  DatatypeSelector getSelector(const std::string& name) const;

  /**
   * Get the name of this datatype.
   * @return The name.
   */
  std::string getName() const;

  /**
   * Get the number of constructors of this datatype.
   * @return The number of constructors.
   */
  size_t getNumConstructors() const;

  /**
   * Get the parameters of this datatype, if it is parametric.
   *
   * @note Asserts that this datatype is parametric.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return The parameters of this datatype.
   */
  std::vector getParameters() const;

  /**
   * Determine if this datatype is parametric.
   * @warning This function is experimental and may change in future versions.
   * @return True if this datatype is parametric.
   */
  bool isParametric() const;

  /**
   * Determine if this datatype corresponds to a co-datatype.
   * @return True if this datatype corresponds to a co-datatype.
   */
  bool isCodatatype() const;

  /**
   * Determine if this datatype corresponds to a tuple.
   * @return True if this datatype corresponds to a tuple.
   */
  bool isTuple() const;

  /**
   * Determine if this datatype corresponds to a record.
   * @warning This function is experimental and may change in future versions.
   * @return True if this datatype corresponds to a record.
   */
  bool isRecord() const;

  /**
   * Determine if a given datatype is finite.
   * @return True if this datatype is finite.
   */
  bool isFinite() const;

  /**
   * Determine if this datatype is well-founded.
   *
   * If this datatype is not a codatatype, this returns false if there are no
   * values of this datatype that are of finite size.
   *
   * @return True if this datatype is well-founded.
   */
  bool isWellFounded() const;

  /**
   * @return True if this Datatype is a null object.
   */
  bool isNull() const;

  /**
   * @return A string representation of this datatype.
   */
  std::string toString() const;

  /**
   * Iterator for the constructors of a datatype.
   */
  class CVC5_EXPORT const_iterator
  {
    friend class Datatype;  // to access constructor

   public:
    /* The following types are required by trait std::iterator_traits */

    /** Iterator tag */
    using iterator_category = std::forward_iterator_tag;

    /** The type of the item */
    using value_type = Datatype;

    /** The pointer type of the item */
    using pointer = const Datatype*;

    /** The reference type of the item */
    using reference = const Datatype&;

    /** The type returned when two iterators are subtracted */
    using difference_type = std::ptrdiff_t;

    /* End of std::iterator_traits required types */

    /** Nullary constructor (required for Cython). */
    const_iterator();

    /**
     * Assignment operator.
     * @param it The iterator to assign to.
     * @return The reference to the iterator after assignment.
     */
    const_iterator& operator=(const const_iterator& it);

    /**
     * Equality operator.
     * @param it The iterator to compare to for equality.
     * @return True if the iterators are equal.
     */
    bool operator==(const const_iterator& it) const;

    /**
     * Disequality operator.
     * @param it The iterator to compare to for disequality.
     * @return True if the iterators are disequal.
     */
    bool operator!=(const const_iterator& it) const;

    /**
     * Increment operator (prefix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator& operator++();

    /**
     * Increment operator (postfix).
     * @return A reference to the iterator after incrementing by one.
     */
    const_iterator operator++(int);

    /**
     * Dereference operator.
     * @return A reference to the constructor this iterator points to.
     */
    const DatatypeConstructor& operator*() const;

    /**
     * Dereference operator.
     * @return A pointer to the constructor this iterator points to.
     */
    const DatatypeConstructor* operator->() const;

   private:
    /**
     * Constructor.
     * @param tm    The associated term manager.
     * @param dtype The internal datatype to iterate over.
     * @param begin True if this is a begin() iterator.
     */
    const_iterator(TermManager* tm, const internal::DType& dtype, bool begin);

    /**
     * The associated term manager.
     */
    TermManager* d_tm = nullptr;

    /**
     * A pointer to the list of constructors of the internal datatype
     * to iterate over.
     * This pointer is maintained for operators == and != only.
     */
    const void* d_int_ctors;

    /** The list of datatype constructor (wrappers) to iterate over. */
    std::vector d_ctors;

    /** The current index of the iterator. */
    size_t d_idx;
  };

  /**
   * @return An iterator to the first constructor of this datatype.
   */
  const_iterator begin() const;

  /**
   * @return An iterator to one-off-the-last constructor of this datatype.
   */
  const_iterator end() const;

 private:
  /**
   * Constructor.
   * @param tm    The associated term manager.
   * @param dtype The internal datatype to be wrapped.
   * @return The Datatype.
   */
  Datatype(TermManager* tm, const internal::DType& dtype);

  /**
   * Return constructor for name.
   * @param name The name of constructor to find.
   * @return The constructor object for the name.
   */
  DatatypeConstructor getConstructorForName(const std::string& name) const;

  /**
   * Return selector for name.
   * @param name The name of selector to find.
   * @return The selector object for the name.
   */
  DatatypeSelector getSelectorForName(const std::string& name) const;

  /**
   * Helper for isNull checks. This prevents calling an API function with
   * CVC5_API_CHECK_NOT_NULL
   */
  bool isNullHelper() const;

  /**
   * The associated term manager.
   */
  TermManager* d_tm = nullptr;

  /**
   * The internal datatype wrapped by this datatype.
   * @note This is a ``std::shared_ptr`` rather than a ``std::unique_ptr``
   *       since ``internal::DType`` is not ref counted.
   */
  std::shared_ptr d_dtype;
};

/**
 * Serialize a datatype declaration to given stream.
 * @param out The output stream.
 * @param dtdecl The datatype declaration to be serialized to the given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out, const DatatypeDecl& dtdecl);

/**
 * Serialize a datatype constructor declaration to given stream.
 * @param out The output stream.
 * @param ctordecl The datatype constructor declaration to be serialized.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out,
                         const DatatypeConstructorDecl& ctordecl);

/**
 * Serialize a vector of datatype constructor declarations to given stream.
 * @param out The output stream.
 * @param vector The vector of datatype constructor declarations to be.
 * serialized to the given stream
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out,
                         const std::vector& vector);

/**
 * Serialize a datatype to given stream.
 * @param out The output stream.
 * @param dtype The datatype to be serialized to given stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Datatype& dtype);

/**
 * Serialize a datatype constructor to given stream.
 * @param out The output stream.
 * @param ctor The datatype constructor to be serialized to given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out, const DatatypeConstructor& ctor);

/**
 * Serialize a datatype selector to given stream.
 * @param out The output stream.
 * @param stor The datatype selector to be serialized to given stream.
 * @return The output stream.
 */
CVC5_EXPORT
std::ostream& operator<<(std::ostream& out, const DatatypeSelector& stor);

}  // namespace cvc5

namespace std {
/**
 * Hash function for datatypes.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Datatype& dt) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Grammar                                                                    */
/* -------------------------------------------------------------------------- */

/**
 * A Sygus Grammar. This class can be used to define a context-free grammar
 * of terms. Its interface coincides with the definition of grammars
 * (``GrammarDef``) in the SyGuS IF 2.1 standard.
 */
class CVC5_EXPORT Grammar
{
  friend class parser::Cmd;
  friend class Solver;
  friend struct std::hash;
  friend std::ostream& operator<<(std::ostream& out, const Grammar& grammar);

 public:
  /**
   * Nullary constructor. Needed for the Cython API.
   */
  Grammar();

  /**
   * Destructor for bookeeping.
   */
  ~Grammar();

  /**
   * Determine if this is the null grammar (Grammar::Grammar()).
   * @return True if this grammar is the null grammar.
   */
  bool isNull() const;

  /**
   * Operator overloading for referential equality of two grammars.
   * @param grammar The grammarto compare to for equality.
   * @return True if both grammars point to the same internal grammar object.
   */
  bool operator==(const Grammar& grammar) const;

  /**
   * Referential disequality operator.
   * @param grammar The grammar to compare to for disequality.
   * @return True if both grammars point to different internal grammar objects.
   */
  bool operator!=(const Grammar& grammar) const;

  /**
   * Add `rule` to the set of rules corresponding to `ntSymbol`.
   * @param ntSymbol The non-terminal to which the rule is added.
   * @param rule The rule to add.
   */
  void addRule(const Term& ntSymbol, const Term& rule);

  /**
   * Add `rules` to the set of rules corresponding to `ntSymbol`.
   * @param ntSymbol The non-terminal to which the rules are added.
   * @param rules The rules to add.
   */
  void addRules(const Term& ntSymbol, const std::vector& rules);

  /**
   * Allow `ntSymbol` to be an arbitrary constant.
   * @param ntSymbol The non-terminal allowed to be any constant.
   */
  void addAnyConstant(const Term& ntSymbol);

  /**
   * Allow `ntSymbol` to be any input variable to corresponding
   * synth-fun/synth-inv with the same sort as `ntSymbol`.
   * @param ntSymbol The non-terminal allowed to be any input variable.
   */
  void addAnyVariable(const Term& ntSymbol);

  /**
   * @return A string representation of this grammar.
   */
  std::string toString() const;

 private:
  /**
   * Constructor.
   * @param tm        The associated term manager.
   * @param sygusVars The input variables to synth-fun/synth-var.
   * @param ntSymbols The non-terminals of this grammar.
   */
  Grammar(TermManager* tm,
          const std::vector& sygusVars,
          const std::vector& ntSymbols);

  /**
   * @return The resolved datatype of the Start symbol of the grammar.
   */
  Sort resolve();

  /**
   * The associated term manager.
   * @note This is only needed temporarily until deprecated term/sort handling
   * functions are removed.
   */
  TermManager* d_tm;
  /** The internal representation of this grammar. */
  std::shared_ptr d_grammar;
};

/**
 * Serialize a grammar to given stream.
 * @param out     The output stream.
 * @param grammar The grammar to be serialized to the given output stream.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& out, const Grammar& grammar);

}  // namespace cvc5

namespace std {
/**
 * Hash function for grammar.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Grammar& grammar) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* Options                                                                    */
/* -------------------------------------------------------------------------- */

/**
 * \verbatim embed:rst:leading-asterisk
 * This class provides type-safe access to a few options that frontends are
 * likely to use, but can not be not be communicated appropriately via the
 * regular :cpp:func:`Solver::getOption() ` or
 * :cpp:func:`Solver::getOptionInfo() `
 * functions. This includes, e.g., the input and output streams that can be
 * configured via :ref:`err `, :ref:`in ` and
 * :ref:`out `. This class does not store the options itself,
 * but only acts as a wrapper to the solver object. It can thus no longer be
 * used after the solver object has been destroyed. \endverbatim
 */
class CVC5_EXPORT DriverOptions
{
  friend class Solver;

 public:
  /** Access the solver's input stream */
  std::istream& in() const;
  /** Access the solver's error output stream */
  std::ostream& err() const;
  /** Access the solver's output stream */
  std::ostream& out() const;

 private:
  DriverOptions(const Solver& solver);
  const Solver& d_solver;
};

/**
 * \verbatim embed:rst:leading-asterisk
 * Holds information about a specific option, including its name, its
 * aliases, whether the option was explicitly set by the user, and information
 * concerning its value. It can be obtained via
 * :cpp:func:`Solver::getOptionInfo() ` and
 * allows for a more detailed inspection of options than
 * :cpp:func:`Solver::getOption() `. The
 * :cpp:member:`valueInfo ` member holds any of the
 * following alternatives:
 *
 * - :cpp:class:`VoidInfo ` if the option holds no
 *   value (or the value has no native type)
 * - :cpp:class:`ValueInfo ` if the option is of
 *   type ``bool`` or ``std::string``, holds the current value and the default
 *   value.
 * - :cpp:class:`NumberInfo ` if the option is of
 *   type ``int64_t``, ``uint64_t`` or ``double``, holds the current and default
 *   value, as well as the minimum and maximum.
 * - :cpp:class:`ModeInfo ` if the option is a mode
 *   option, holds the current and default values, as well as a list of valid
 *   modes.
 *
 * Additionally, this class provides convenience functions to obtain the
 * current value of an option in a type-safe manner using :cpp:func:`boolValue()
 * `, :cpp:func:`stringValue()
 * `, :cpp:func:`intValue()
 * `, :cpp:func:`uintValue()
 * ` and :cpp:func:`doubleValue()
 * `. They assert that the option has the
 * respective type and return the current value.
 *
 * If the option has a special type that is not covered by the above
 * alternatives, the :cpp:member:`valueInfo ` holds
 * a :cpp:class:`VoidInfo `. Some options, that are
 * expected to be used by frontends (e.g., input and output streams) can also
 * be accessed using :cpp:func:`Solver::getDriverOptions()
 * `. \endverbatim
 */
struct CVC5_EXPORT OptionInfo
{
  /** Has no value information. */
  struct VoidInfo
  {
  };
  /** Basic information for option values. ``T`` can be ``bool`` or
   * ``std::string``. */
  template 
  struct ValueInfo
  {
    /** The default value. */
    T defaultValue;
    /** The current value. */
    T currentValue;
  };
  /** Information for numeric values. ``T`` can be ``int64_t``, ``uint64_t`` or
   * ``double``. */
  template 
  struct NumberInfo
  {
    /** The default value. */
    T defaultValue;
    /** The current value. */
    T currentValue;
    /** The optional minimum value. */
    std::optional minimum;
    /** The optional maximum value. */
    std::optional maximum;
  };
  /** Information for mode option values. */
  struct ModeInfo
  {
    /** The default value. */
    std::string defaultValue;
    /** The current value. */
    std::string currentValue;
    /** The possible modes. */
    std::vector modes;
  };

  /** The option name */
  std::string name;
  /** The option name aliases */
  std::vector aliases;
  /** Whether the option was explicitly set by the user */
  bool setByUser;
  /** Whether this is an expert option */
  bool isExpert;
  /** Whether this is a regular option */
  bool isRegular;
  /** Possible types for ``valueInfo``. */
  using OptionInfoVariant = std::variant,
                                         ValueInfo,
                                         NumberInfo,
                                         NumberInfo,
                                         NumberInfo,
                                         ModeInfo>;
  /** The option value information */
  OptionInfoVariant valueInfo;
  /**
   * Get the current value as a `bool`.
   * @note Asserts that `valueInfo` holds a `bool`.
   * @return The current value as a `bool`.
   */
  bool boolValue() const;
  /**
   * Get the current value as a `std::string`.
   * @note Asserts that `valueInfo` holds a `std::string`.
   * @return The current value as a `std::string`.
   */
  std::string stringValue() const;
  /**
   * Get the current value as an `int64_t`.
   * @note Asserts that `valueInfo` holds an `int64_t`.
   * @return The current value as a `int64_t`.
   */
  int64_t intValue() const;
  /**
   * Get the current value as a `uint64_t`.
   * @note Asserts that `valueInfo` holds a `uint64_t`.
   * @return The current value as a `uint64_t`.
   */
  uint64_t uintValue() const;
  /**
   * Obtain the current value as a `double`.
   * @note Asserts that `valueInfo` holds a `double`.
   * @return The current value as a `double`.
   */
  double doubleValue() const;
  /**
   * Get a string representation of an option info.
   * @return The string representation.
   */
  std::string toString() const;
};

/**
 * Print an `OptionInfo` object to an output stream.
 * @param os The output stream.
 * @param oi The option info.
 * @return The output stream.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& os, const OptionInfo& oi);

/* -------------------------------------------------------------------------- */
/* Statistics                                                                 */
/* -------------------------------------------------------------------------- */

/**
 * \verbatim embed:rst:leading-asterisk
 * Represents a snapshot of a single statistic value. See :doc:`/statistics` for
 * how statistics can be used.
 * A value can be of type ``int64_t``, ``double``, ``std::string`` or a
 * histogram
 * (``std::map``).
 * The value type can be queried (using ``isInt()``, ``isDouble()``, etc.) and
 * the stored value can be accessed (using ``getInt()``, ``getDouble()``, etc.).
 * It is possible to query whether this statistic is an internal statistic by
 * :cpp:func:`isInternal() ` and whether its value is
 * the default value by :cpp:func:`isDefault() `.
 * \endverbatim
 */
class CVC5_EXPORT Stat
{
  struct StatData;

 public:
  friend class Statistics;
  CVC5_EXPORT friend std::ostream& operator<<(std::ostream& os, const Stat& sv);
  /** Representation of a histogram: maps names to frequencies. */
  using HistogramData = std::map;
  /**
   * Create an empty statistics object. On such an object all ``isX()`` return
   * false and all ``getX()`` throw an API exception. It solely exists because
   * it makes implementing bindings for other languages much easier.
   */
  Stat();
  /** Copy constructor */
  Stat(const Stat& s);
  /** Destructor */
  ~Stat();
  /** Copy assignment */
  Stat& operator=(const Stat& s);

  /**
   * Determine if this statistic is intended for internal use only.
   * @return True if this is an internal statistic.
   */
  bool isInternal() const;
  /**
   * Determine if this statistic holds the default value.
   * @return True if this is a defaulted statistic.
   */
  bool isDefault() const;

  /**
   * Determine if this statistic holds an integer value.
   * @return True if this value is an integer.
   */
  bool isInt() const;
  /**
   * Get the value of an integer statistic.
   * @return The integer value.
   */
  int64_t getInt() const;
  /**
   * Determine if this statistic holds a double value.
   * @return True if this value is a double.
   */
  bool isDouble() const;
  /**
   * Get the value of a double statistic.
   * @return The double value.
   */
  double getDouble() const;
  /**
   * Determine if this statistic holds a string value.
   * @return True if this value is a string.
   */
  bool isString() const;
  /**
   * Get the value of a string statistic.
   * @return The string value.
   */
  const std::string& getString() const;
  /**
   * Determine if this statistics holds a histogram.
   * @return True if this value is a histogram.
   */
  bool isHistogram() const;
  /**
   * Get the value of a histogram statistic.
   * @return The histogram value.
   */
  const HistogramData& getHistogram() const;

  /**
   * Get a string represenation of this statistic.
   * @return The string represenation.
   */
  std::string toString() const;

 private:
  Stat(bool internal, bool def, StatData&& sd);
  /** Whether this statistic is only meant for internal use */
  bool d_internal;
  /** Whether this statistic has the default value */
  bool d_default;
  std::unique_ptr d_data;
};

/**
 * Print a `Stat` object to an ``std::ostream``.
 */
CVC5_EXPORT std::ostream& operator<<(std::ostream& os, const Stat& stat);

/**
 * \verbatim embed:rst:leading-asterisk
 * Represents a snapshot of the solver statistics. See :doc:`/statistics` for
 * how statistics can be used.
 *
 * Statistics can be queried from the Solver via
 * :cpp:func:`Solver::getStatistics() `, and
 * from the TermManager via :cpp:func:`TermManager::getStatistics()
 * `. An statistics instance obtained from
 * either call is independent of the :cpp:class:`Solver ` (and
 * its associated :cpp:class:`TermManager `object: it will
 * not change when new terms are created or the solver's internal statistics
 * do. It will also not be invalidated if the solver/term manageris destroyed.
 *
 * Iterating over this class (via :cpp:func:`begin()
 * ` and :cpp:func:`end() `)
 * shows only public statistics that have been changed. By passing appropriate
 * flags to :cpp:func:`begin() `, statistics that are
 * internal, defaulted, or both, can be included as well. A single statistic
 * value is represented as :cpp:class:`Stat `. \endverbatim
 */
class CVC5_EXPORT Statistics
{
 public:
  friend class Solver;
  friend class TermManager;
  /** How the statistics are stored internally. */
  using BaseType = std::map;

  /** Custom iterator to hide certain statistics from regular iteration */
  class CVC5_EXPORT iterator
  {
   public:
    friend class Statistics;
    iterator() = default;
    BaseType::const_reference operator*() const;
    BaseType::const_pointer operator->() const;
    iterator& operator++();
    iterator operator++(int);
    iterator& operator--();
    iterator operator--(int);
    bool operator==(const iterator& rhs) const;
    bool operator!=(const iterator& rhs) const;

   private:
    iterator(BaseType::const_iterator it,
             const BaseType& base,
             bool internal,
             bool defaulted);
    bool isVisible() const;
    BaseType::const_iterator d_it;
    const BaseType* d_base;
    bool d_showInternal = false;
    bool d_showDefault = false;
  };

  /** Creates an empty statistics object. */
  Statistics() = default;

  /**
   * Retrieve the statistic with the given name.
   * @note Asserts that a statistic with the given name actually exists and
   *       throws a CVC5ApiRecoverableException if it does not.
   * @param name The name of the statistic.
   * @return The statistic with the given name.
   */
  const Stat& get(const std::string& name);
  /**
   * Begin iteration over the statistics values.
   * By default, only entries that are public and have been set
   * are visible while the others are skipped.
   * @param internal If set to true, internal statistics are shown as well.
   * @param defaulted If set to true, defaulted statistics are shown as well.
   */
  iterator begin(bool internal = true, bool defaulted = true) const;
  /** End iteration */
  iterator end() const;

  /**
   * Get a string represenation of this statistics object.
   * @return The string represenation.
   */
  std::string toString() const;

 private:
  Statistics(const internal::StatisticsRegistry& reg);
  /** Internal data */
  BaseType d_stats;
};
CVC5_EXPORT std::ostream& operator<<(std::ostream& out,
                                     const Statistics& stats);

/* -------------------------------------------------------------------------- */
/* Plugin                                                                     */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 plugin.
 */
class CVC5_EXPORT Plugin
{
  friend class Solver;

 public:
  Plugin(TermManager& tm);
  virtual ~Plugin() = default;
  /**
   * Call to check, return vector of lemmas to add to the SAT solver.
   * This method is called periodically, roughly at every SAT decision.
   *
   * @return The vector of lemmas to add to the SAT solver.
   */
  virtual std::vector check();
  /**
   * Notify SAT clause, called when `clause` is learned by the SAT solver.
   * @param clause The learned clause.
   */
  virtual void notifySatClause(const Term& clause);
  /**
   * Notify theory lemma, called when `lemma` is sent by a theory solver.
   * @param lemma The theory lemma.
   */
  virtual void notifyTheoryLemma(const Term& lemma);
  /**
   * Get the name of the plugin (for debugging).
   * @return The name of the plugin.
   */
  virtual std::string getName() = 0;

 private:
  /** Converter to external */
  std::shared_ptr d_pExtToInt;
};

/* -------------------------------------------------------------------------- */
/* Proof                                                                      */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 proof.  Proofs are trees and every proof object corresponds to the
 * root step of a proof.  The branches of the root step are the premises of
 * the step.
 */
class CVC5_EXPORT Proof
{
  friend class Solver;
  friend struct std::hash;

 public:
  /**
   * Nullary constructor. Needed for the Cython API.
   */
  Proof();
  /*
   * Destructor.
   */
  ~Proof();

  /**
   * Determine if this is the null proof (Proof::Proof()).
   * @return True if this grammar is the null proof.
   */
  bool isNull() const;

  /**
   * Get the proof rule used by the root step of the proof.
   * @return The proof rule.
   */
  ProofRule getRule() const;

  /**
   * Get the proof rewrite rule used  by the root step of the proof.
   *
   * Requires that `getRule()` does not return #DSL_REWRITE or
   * #THEORY_REWRITE.
   *
   * @return The proof rewrite rule.
   *
   */
  ProofRewriteRule getRewriteRule() const;

  /** @return The conclusion of the root step of the proof. */
  Term getResult() const;

  /** @return The premises of the root step of the proof. */
  const std::vector getChildren() const;

  /**
   * @return The arguments of the root step of the proof as a vector of terms.
   *         Some of those terms might be strings.
   */
  const std::vector getArguments() const;

  /**
   * Operator overloading for referential equality of two proofs.
   * @param p The proof to compare to for equality.
   * @return True if both proofs point to the same internal proof object.
   */
  bool operator==(const Proof& p) const;

  /**
   * Referential disequality operator.
   * @param p The proof to compare to for disequality.
   * @return True if both proofs point to different internal proof objects.
   */
  bool operator!=(const Proof& p) const;

 private:
  /** Construct a proof by wrapping a ProofNode. */
  Proof(TermManager* tm, const std::shared_ptr p);

  /** The internal proof node wrapped by this proof object. */
  std::shared_ptr d_proofNode;
  /**
   * The associated term manager.
   * @note This is only needed temporarily until deprecated term/sort handling
   * functions are removed.
   */
  TermManager* d_tm;
};

}  // namespace cvc5

namespace std {
/**
 * Hash function for proofs.
 */
template <>
struct CVC5_EXPORT hash
{
  size_t operator()(const cvc5::Proof& p) const;
};
}  // namespace std

namespace cvc5 {

/* -------------------------------------------------------------------------- */
/* TermManager                                                                */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 term manager.
 */
class CVC5_EXPORT TermManager
{
  friend class Sort;
  friend class Op;
  friend class Term;
  friend class DatatypeConstructorDecl;
  friend class DatatypeDecl;
  friend class Grammar;
  friend class Plugin;
  friend class Solver;

 public:
  /** Constructor. */
  TermManager();
  /** Destructor. */
  ~TermManager();

  /**
   * Get a snapshot of the current state of the statistic values of this
   * term manager.
   *
   * Term manager statistics are independent from any solver instance. The
   * returned object is completely decoupled from the term manager and will
   * not change when the solver is used again.
   *
   * @return A snapshot of the current state of the statistic values.
   */
  Statistics getStatistics() const;

  /**
   * Print the statistics to the given file descriptor, suitable for usage in
   * signal handlers.
   * @param fd The file descriptor.
   */
  void printStatisticsSafe(int fd) const;

  /* Sorts -------------------------------------------------------------- */

  /**
   * Get the Boolean sort.
   * @return Sort `Bool`.
   */
  Sort getBooleanSort();
  /**
   * Get the Integer sort.
   * @return Sort `Int`.
   */
  Sort getIntegerSort();
  /**
   * Get the Real sort.
   * @return Sort `Real`.
   */
  Sort getRealSort();
  /**
   * Get the regular expression sort.
   * @return Sort `RegExp`.
   */
  Sort getRegExpSort();
  /**
   * Get the rounding mode sort.
   * @return Sort `RoundingMode`.
   */
  Sort getRoundingModeSort();
  /**
   * Get the string sort.
   * @return Sort `String`.
   */
  Sort getStringSort();
  /**
   * Create an array sort.
   * @param indexSort The array index sort.
   * @param elemSort  The array element sort.
   * @return The array sort.
   */
  Sort mkArraySort(const Sort& indexSort, const Sort& elemSort);
  /**
   * Create a bit-vector sort.
   * @param size The bit-width of the bit-vector sort.
   * @return The bit-vector sort.
   */
  Sort mkBitVectorSort(uint32_t size);
  /**
   * Create a floating-point sort.
   * @param exp The bit-width of the exponent of the floating-point sort.
   * @param sig The bit-width of the significand of the floating-point sort.
   * @return The floating-point sort.
   */
  Sort mkFloatingPointSort(uint32_t exp, uint32_t sig);
  /**
   * Create a finite-field sort from a given string of base n.
   * @param size The modulus of the field. Must be prime.
   * @param base The base of the string representation of `size`.
   * @return The finite-field sort.
   */
  Sort mkFiniteFieldSort(const std::string& size, uint32_t base = 10);
  /**
   * Create a datatype sort.
   * @param dtypedecl The datatype declaration from which the sort is created.
   * @return The datatype sort.
   */
  Sort mkDatatypeSort(const DatatypeDecl& dtypedecl);
  /**
   * Create a vector of datatype sorts.
   *
   * @note  The names of the datatype declarations must be distinct.
   *
   * @param dtypedecls The datatype declarations from which the sort is created.
   * @return The datatype sorts.
   */
  std::vector mkDatatypeSorts(
      const std::vector& dtypedecls);
  /**
   * Create function sort.
   * @param sorts    The sort of the function arguments.
   * @param codomain The sort of the function return value.
   * @return The function sort.
   */
  Sort mkFunctionSort(const std::vector& sorts, const Sort& codomain);
  /**
   * Create a skolem.
   * @param id      The skolem identifier.
   * @param indices The indices of the skolem.
   * @return The skolem.
   */
  Term mkSkolem(SkolemId id, const std::vector& indices);
  /**
   * Get the number of indices for a skolem id.
   * @param id The skolem id.
   * @return The number of indices for the skolem id.
   */
  size_t getNumIndicesForSkolemId(SkolemId id);
  /**
   * Create a sort parameter.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param symbol The name of the sort.
   * @return The sort parameter.
   */
  Sort mkParamSort(const std::optional& symbol = std::nullopt);
  /**
   * Create a predicate sort.
   *
   * This is equivalent to calling mkFunctionSort() with the Boolean sort as the
   * codomain.
   *
   * @param sorts The list of sorts of the predicate.
   * @return The predicate sort.
   */
  Sort mkPredicateSort(const std::vector& sorts);
  /**
   * Create a record sort
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param fields The list of fields of the record.
   * @return The record sort.
   */
  Sort mkRecordSort(const std::vector>& fields);
  /**
   * Create a set sort.
   * @param elemSort The sort of the set elements.
   * @return The set sort.
   */
  Sort mkSetSort(const Sort& elemSort);
  /**
   * Create a bag sort.
   * @param elemSort The sort of the bag elements.
   * @return The bag sort.
   */
  Sort mkBagSort(const Sort& elemSort);
  /**
   * Create a sequence sort.
   * @param elemSort The sort of the sequence elements.
   * @return The sequence sort.
   */
  Sort mkSequenceSort(const Sort& elemSort);
  /**
   * Create an abstract sort. An abstract sort represents a sort for a given
   * kind whose parameters and arguments are unspecified.
   *
   * The kind `k` must be the kind of a sort that can be abstracted, i.e., a
   * sort that has indices or argument sorts. For example, #ARRAY_SORT and
   * #BITVECTOR_SORT can be passed as the kind `k` to this function, while
   * #INTEGER_SORT and #STRING_SORT cannot.
   *
   * @note Providing the kind #ABSTRACT_SORT as an argument to this function
   * returns the (fully) unspecified sort, denoted `?`.
   *
   * @note Providing a kind `k` that has no indices and a fixed arity
   * of argument sorts will return the sort of kind `k` whose arguments are the
   * unspecified sort. For example, `mkAbstractSort(SortKind::ARRAY_SORT)` will
   * return the sort `(ARRAY_SORT ? ?)` instead of the abstract sort whose
   * abstract kind is #ARRAY_SORT.
   *
   * @param k The kind of the abstract sort
   * @return The abstract sort.
   *
   * @warning This function is experimental and may change in future versions.
   */
  Sort mkAbstractSort(SortKind k);
  /**
   * Create an uninterpreted sort.
   * @param symbol The name of the sort.
   * @return The uninterpreted sort.
   */
  Sort mkUninterpretedSort(
      const std::optional& symbol = std::nullopt);
  /**
   * Create an unresolved datatype sort.
   *
   * This is for creating yet unresolved sort placeholders for mutually
   * recursive parametric datatypes.
   *
   * @param symbol The symbol of the sort.
   * @param arity The number of sort parameters of the sort.
   * @return The unresolved sort.
   *
   * @warning This function is experimental and may change in future versions.
   */
  Sort mkUnresolvedDatatypeSort(const std::string& symbol, size_t arity = 0);
  /**
   * Create an uninterpreted sort constructor sort.
   *
   * An uninterpreted sort constructor is an uninterpreted sort with arity > 0.
   *
   * @param symbol The symbol of the sort.
   * @param arity The arity of the sort (must be > 0)
   * @return The uninterpreted sort constructor sort.
   */
  Sort mkUninterpretedSortConstructorSort(
      size_t arity, const std::optional& symbol = std::nullopt);
  /**
   * Create a tuple sort.
   * @param sorts The sorts of the elements of the tuple.
   * @return The tuple sort.
   */
  Sort mkTupleSort(const std::vector& sorts);
  /**
   * Create a nullable sort.
   * @param sort The sort of the element of the nullable.
   * @return The nullable sort.
   */
  Sort mkNullableSort(const Sort& sort);

  /* Operators ---------------------------------------------------------- */

  /**
   * Create operator of Kind:
   *   - #BITVECTOR_EXTRACT
   *   - #BITVECTOR_REPEAT
   *   - #BITVECTOR_ROTATE_LEFT
   *   - #BITVECTOR_ROTATE_RIGHT
   *   - #BITVECTOR_SIGN_EXTEND
   *   - #BITVECTOR_ZERO_EXTEND
   *   - #DIVISIBLE
   *   - #FLOATINGPOINT_TO_FP_FROM_FP
   *   - #FLOATINGPOINT_TO_FP_FROM_IEEE_BV
   *   - #FLOATINGPOINT_TO_FP_FROM_REAL
   *   - #FLOATINGPOINT_TO_FP_FROM_SBV
   *   - #FLOATINGPOINT_TO_FP_FROM_UBV
   *   - #FLOATINGPOINT_TO_SBV
   *   - #FLOATINGPOINT_TO_UBV
   *   - #INT_TO_BITVECTOR
   *   - #TUPLE_PROJECT
   *
   * See cvc5::Kind for a description of the parameters.
   * @param kind The kind of the operator.
   * @param args The arguments (indices) of the operator.
   *
   * @note If ``args`` is empty, the Op simply wraps the cvc5::Kind.  The
   * Kind can be used in Solver::mkTerm directly without creating an Op
   * first.
   */
  Op mkOp(Kind kind, const std::vector& args = {});
#ifndef DOXYGEN_SKIP
  // Overload is only used to disambiguate the std::vector and std::string
  // overloads.
  Op mkOp(Kind kind, const std::initializer_list& args);
#endif
  /**
   * Create operator of kind:
   *   - #DIVISIBLE (to support arbitrary precision integers)
   * See cvc5::Kind for a description of the parameters.
   * @param kind The kind of the operator.
   * @param arg The string argument to this operator.
   */
  Op mkOp(Kind kind, const std::string& arg);

  /* Terms -------------------------------------------------------------- */

  /**
   * Create n-ary term of given kind.
   * @param kind     The kind of the term.
   * @param children The children of the term.
   * @return The term.
   */
  Term mkTerm(Kind kind, const std::vector& children = {});
  /**
   * Create n-ary term of given kind from a given operator.
   *
   * Create operators with mkOp().
   *
   * @param op       The operator.
   * @param children The children of the term.
   * @return The Term.
   */
  Term mkTerm(const Op& op, const std::vector& children = {});

  /* Constants, Values and Special Terms -------------------------------- */

  /**
   * Create a Boolean true constant.
   * @return The `true` constant.
   */
  Term mkTrue();
  /**
   * Create a Boolean false constant.
   * @return The `false` constant.
   */
  Term mkFalse();
  /**
   * Create a Boolean constant.
   * @return The Boolean constant.
   * @param val The value of the constant.
   */
  Term mkBoolean(bool val);
  /**
   * Create a constant representing the number Pi.
   * @return A constant representing Pi.
   */
  Term mkPi();
  /**
   * Create an integer constant from a string.
   * @param s The string representation of the constant, may represent an
   *          integer (e.g., "123").
   * @return A constant of sort Integer assuming `s` represents an integer)
   */
  Term mkInteger(const std::string& s);
  /**
   * Create an integer constant from a c++ int.
   * @param val The value of the constant.
   * @return A constant of sort Integer.
   */
  Term mkInteger(int64_t val);
  /**
   * Create a real constant from a string.
   * @param s The string representation of the constant, may represent an
   *          integer (e.g., "123") or real constant (e.g., "12.34" or "12/34").
   * @return A constant of sort Real.
   */
  Term mkReal(const std::string& s);
  /**
   * Create a real constant from an integer.
   * @param val The value of the constant.
   * @return A constant of sort Integer.
   */
  Term mkReal(int64_t val);
  /**
   * Create a real constant from a rational.
   * @param num The value of the numerator.
   * @param den The value of the denominator.
   * @return A constant of sort Real.
   */
  Term mkReal(int64_t num, int64_t den);
  /**
   * Create a regular expression all (re.all) term.
   * @return The all term.
   */
  Term mkRegexpAll();
  /**
   * Create a regular expression allchar (re.allchar) term.
   * @return The allchar term.
   */
  Term mkRegexpAllchar();
  /**
   * Create a regular expression none (re.none) term.
   * @return The none term.
   */
  Term mkRegexpNone();
  /**
   * Create a constant representing an empty set of the given sort.
   * @param sort The sort of the set elements.
   * @return The empty set constant.
   */
  Term mkEmptySet(const Sort& sort);
  /**
   * Create a constant representing an empty bag of the given sort.
   * @param sort The sort of the bag elements.
   * @return The empty bag constant.
   */
  Term mkEmptyBag(const Sort& sort);
  /**
   * Create a separation logic empty term.
   * @return The separation logic empty term.
   * @warning This function is experimental and may change in future versions.
   */
  Term mkSepEmp();
  /**
   * Create a separation logic nil term.
   * @param sort The sort of the nil term.
   * @return The separation logic nil term.
   * @warning This function is experimental and may change in future versions.
   */
  Term mkSepNil(const Sort& sort);
  /**
   * Create a String constant from a `std::string` which may contain SMT-LIB
   * compatible escape sequences like `\u1234` to encode unicode characters.
   * @param s               The string this constant represents.
   * @param useEscSequences Determines whether escape sequences in `s` should
   *                        be converted to the corresponding unicode character.
   * @return The String constant.
   */
  Term mkString(const std::string& s, bool useEscSequences = false);
  /**
   * Create a String constant from a `std::wstring`.
   *
   * This function does not support escape sequences as `std::wstring` already
   * supports unicode characters.
   *
   * @param s The string this constant represents.
   * @return The String constant.
   */
  Term mkString(const std::wstring& s);
  /**
   * Create an empty sequence of the given element sort.
   * @param sort The element sort of the sequence.
   * @return The empty sequence with given element sort.
   */
  Term mkEmptySequence(const Sort& sort);
  /**
   * Create a universe set of the given sort.
   * @param sort The sort of the set elements.
   * @return The universe set constant.
   */
  Term mkUniverseSet(const Sort& sort);
  /**
   * Create a bit-vector constant of given size and value.
   *
   * @note The given value must fit into a bit-vector of the given size.
   *
   * @param size The bit-width of the bit-vector sort.
   * @param val  The value of the constant.
   * @return The bit-vector constant.
   */
  Term mkBitVector(uint32_t size, uint64_t val = 0);
  /**
   * Create a bit-vector constant of a given bit-width from a given string of
   * base 2, 10 or 16.
   *
   * @note The given value must fit into a bit-vector of the given size.
   *
   * @param size The bit-width of the constant.
   * @param s    The string representation of the constant.
   * @param base The base of the string representation (`2` for binary, `10`
   *             for decimal, and `16` for hexadecimal).
   * @return The bit-vector constant.
   */
  Term mkBitVector(uint32_t size, const std::string& s, uint32_t base);
  /**
   * Create a finite field constant in a given field from a given string
   * of base n.
   *
   * If `size` is the field size, the constant needs not be in the range
   * [0,size). If it is outside this range, it will be reduced modulo size
   * before being constructed.
   *
   * @param value The string representation of the constant.
   * @param sort  The field sort.
   * @param base  The base of the string representation of `value`.
   *
   */
  Term mkFiniteFieldElem(const std::string& value,
                         const Sort& sort,
                         uint32_t base = 10);
  /**
   * Create a constant array with the provided constant value stored at every
   * index.
   * @param sort The sort of the constant array (must be an array sort).
   * @param val  The constant value to store (must match the sort's element
   *             sort).
   * @return The constant array term.
   */
  Term mkConstArray(const Sort& sort, const Term& val);
  /**
   * Create a positive infinity floating-point constant (SMT-LIB: `+oo`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   */
  Term mkFloatingPointPosInf(uint32_t exp, uint32_t sig);
  /**
   * Create a negative infinity floating-point constant (SMT-LIB: `-oo`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   */
  Term mkFloatingPointNegInf(uint32_t exp, uint32_t sig);
  /**
   * Create a not-a-number floating-point constant (SMT-LIB: `NaN`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   */
  Term mkFloatingPointNaN(uint32_t exp, uint32_t sig);
  /**
   * Create a positive zero floating-point constant (SMT-LIB: +zero).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   */
  Term mkFloatingPointPosZero(uint32_t exp, uint32_t sig);
  /**
   * Create a negative zero floating-point constant (SMT-LIB: -zero).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   */
  Term mkFloatingPointNegZero(uint32_t exp, uint32_t sig);
  /**
   * Create a rounding mode value.
   * @param rm The floating point rounding mode this constant represents.
   * @return The rounding mode value.
   */
  Term mkRoundingMode(RoundingMode rm);
  /**
   * Create a floating-point value from a bit-vector given in IEEE-754
   * format.
   * @param exp Size of the exponent.
   * @param sig Size of the significand.
   * @param val Value of the floating-point constant as a bit-vector term.
   * @return The floating-point value.
   */
  Term mkFloatingPoint(uint32_t exp, uint32_t sig, const Term& val);
  /**
   * Create a floating-point value from its three IEEE-754 bit-vector
   * value components (sign bit, exponent, significand).
   * @param sign The sign bit.
   * @param exp  The bit-vector representing the exponent.
   * @param sig  The bit-vector representing the significand.
   * @return The floating-point value.
   */
  Term mkFloatingPoint(const Term& sign, const Term& exp, const Term& sig);
  /**
   * Create a cardinality constraint for an uninterpreted sort.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param sort       The sort the cardinality constraint is for.
   * @param upperBound The upper bound on the cardinality of the sort.
   * @return The cardinality constraint.
   */
  Term mkCardinalityConstraint(const Sort& sort, uint32_t upperBound);
  /**
   * Create a tuple term.
   * @param terms The elements in the tuple.
   * @return The tuple Term.
   */
  Term mkTuple(const std::vector& terms);
  /**
   * Create a nullable some term.
   * @param term The element value.
   * @return the Element value wrapped in some constructor.
   */
  Term mkNullableSome(const Term& term);
  /**
   * Create a selector for nullable term.
   * @param term A nullable term.
   * @return The element value of the nullable term.
   */
  Term mkNullableVal(const Term& term);
  /**
   * Create a null tester for a nullable term.
   * @param term A nullable term.
   * @return A tester whether term is null.
   */
  Term mkNullableIsNull(const Term& term);
  /**
   * Create a some tester for a nullable term.
   * @param term A nullable term.
   * @return A tester whether term is some.
   */
  Term mkNullableIsSome(const Term& term);
  /**
   * Create a constant representing an null of the given sort.
   * @param sort The sort of the Nullable element.
   * @return The null constant.
   */
  Term mkNullableNull(const Sort& sort);
  /**
   * Create a term that lifts kind to nullable terms.
   *
   * Example:
   * If we have the term ((_ nullable.lift +) x y),
   * where x, y of type (Nullable Int), then
   * kind would be ADD, and args would be [x, y].
   * This function would return
   * (nullable.lift (lambda ((a Int) (b Int)) (+ a b)) x y)
   *
   * @param kind The lifted operator.
   * @param args The arguments of the lifted operator.
   * @return A term of Kind NULLABLE_LIFT where the first child is a lambda
   *         expression, and the remaining children are the original arguments.
   */
  Term mkNullableLift(Kind kind, const std::vector& args);

  /* Constants and Variables -------------------------------------------- */

  /**
   * Create a free constant.
   *
   * Note that the returned term is always fresh, even if the same arguments
   * were provided on a previous call to mkConst().
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-const  )
   *     (declare-fun  () )
   * \endverbatim
   *
   * @param sort   The sort of the constant.
   * @param symbol The name of the constant (optional).
   * @return The constant.
   */
  Term mkConst(const Sort& sort,
               const std::optional& symbol = std::nullopt);

  /**
   * Create a bound variable to be used in a binder (i.e., a quantifier, a
   * lambda, or a witness binder).
   *
   * @note The returned term is always fresh, even if the same arguments
   *       were provided on a previous call to mkConst().
   *
   * @param sort   The sort of the variable.
   * @param symbol The name of the variable (optional).
   * @return The variable.
   */
  Term mkVar(const Sort& sort,
             const std::optional& symbol = std::nullopt);

  /* Datatype Constructor Declaration ----------------------------------- */

  /**
   * Create a datatype constructor declaration.
   * @param name The name of the datatype constructor.
   * @return The DatatypeConstructorDecl.
   */
  DatatypeConstructorDecl mkDatatypeConstructorDecl(const std::string& name);

  /* Datatype Declaration ----------------------------------------------- */

  /**
   * Create a datatype declaration.
   * @param name         The name of the datatype.
   * @param isCoDatatype True if a codatatype is to be constructed.
   * @return The DatatypeDecl.
   */
  DatatypeDecl mkDatatypeDecl(const std::string& name,
                              bool isCoDatatype = false);

  /**
   * Create a datatype declaration.
   *
   * Create sorts parameter with Solver::mkParamSort().
   *
   * @param name         The name of the datatype.
   * @param params       A list of sort parameters.
   * @param isCoDatatype True if a codatatype is to be constructed.
   * @return The DatatypeDecl.
   * @warning This function is experimental and may change in future versions.
   */
  DatatypeDecl mkDatatypeDecl(const std::string& name,
                              const std::vector& params,
                              bool isCoDatatype = false);

 private:
  /**
   * Get the current, thread_local term manager.
   * @warning This is a workaround to provide deprecated functionality,
   *          specifically, constructor Solver::Solver(). When this constructor
   *          is removed, a solver instance MUST be constructed from a given
   *          TermManager instance, and this workaround will be removed.
   */
  static TermManager* currentTM();

  /** Reset the API statistics. */
  void resetStatistics();
  /** Increment the term stats counter. */
  void increment_term_stats(Kind kind) const;
  /** Increment the vars stats or consts stats counter. */
  void increment_vars_consts_stats(const internal::TypeNode& type,
                                   bool is_var) const;

  /** Helper to check for API misuse in mkOp functions. */
  void checkMkTerm(Kind kind, uint32_t nchildren) const;
  /**
   * Check whether string s is a valid decimal integer.
   * @param s The string representation to check.
   * @return True if `s` is a valid decimal integer.
   */
  bool isValidInteger(const std::string& s) const;

  /**
   * Helper for calls to mkVar from the TermManager and Solver. Ensures that
   * API statistics are collected.
   * @param type   The internal type of the variable.
   * @param symbol The symbol of the variable.
   */
  internal::Node mkVarHelper(
      const internal::TypeNode& type,
      const std::optional& symbol = std::nullopt);
  /**
   * Helper for calls to mkConst from the TermManager and Solver. Ensures that
   * API statistics are collected.
   * @param type   The internal type of the const.
   * @param symbol The symbol of the const.
   * @param fresh  True to return a fresh variable. If false, it returns the
   *               same variable for the given type and name.
   */
  internal::Node mkConstHelper(const internal::TypeNode& type,
                               const std::optional& symbol,
                               bool fresh = true);
  /**
   * Helper for mk-functions that call NodeManager::mkConst().
   * @param t  The value.
   * @return The value term.
   */
  template 
  Term mkValHelper(const T& t);
  /** Helper for creating operators. */
  template 
  Op mkOpHelper(Kind kind, const T& t);
  /**
   * Helper for creating rational values.
   * @param r  The value (either int or real).
   * @param    isInt True to create an integer value.
   * @return The rational value term.
   */
  Term mkRationalValHelper(const internal::Rational& r, bool isInt);
  /**
   * Helper for mkReal functions that take a string as argument.
   * @param s     The string representation of the real/int value.
   * @param isInt True if `s` represents an integer.
   * @return The value term.
   */
  Term mkRealOrIntegerFromStrHelper(const std::string& s, bool isInt = true);
  /**
   * Helper for mkBitVector functions that take a string as argument.
   * @param s    The string representation of the bit-vector value.
   * @param base The numerical base of the string representation.
   * @return The bit-vector value term.
   */
  Term mkBVFromStrHelper(const std::string& s, uint32_t base);
  /**
   * Helper for mkBitVector functions that take a string and a size as
   * arguments.
   * @param size The bit-vector size.
   * @param s    The string representation of the bit-vector value.
   * @param base The numerical base of the string representation.
   * @return The bit-vector value term.
   */
  Term mkBVFromStrHelper(uint32_t size, const std::string& s, uint32_t base);
  /**
   * Helper for mkBitVector functions that take an unsigned integer as argument.
   * @param size The bit-vector size.
   * @param val  The bit-vector value as uint64_t.
   * @return The bit-vector value term.
   */
  Term mkBVFromIntHelper(uint32_t size, uint64_t val);
  /**
   * Helper for functions that create tuple sorts.
   * @param sorts The sorts.
   * @return The tuple sort.
   */
  Sort mkTupleSortHelper(const std::vector& sorts);
  /**
   * Helper for mkTerm functions that create Term from a Kind.
   * @param kind The kind.
   * @return The term.
   */
  Term mkTermFromKind(Kind kind);
  /**
   * Helper for mkChar functions that take a string as argument.
   * @param s The string.
   * @return The character term.
   */
  Term mkCharFromStrHelper(const std::string& s);
  /**
   * Create n-ary term of given kind. This handles the cases of left/right
   * associative operators, chainable operators, and cases when the number of
   * children exceeds the maximum arity for the kind.
   * @param kind     The kind of the term.
   * @param children The children of the term.
   * @return The Term.
   */
  Term mkTermHelper(Kind kind, const std::vector& children);
  /**
   * Create n-ary term of given kind from a given operator.
   * @param op       The operator.
   * @param children The children of the term.
   * @return The Term.
   */
  Term mkTermHelper(const Op& op, const std::vector& children);

  /** The associated node manager. */
  internal::NodeManager* d_nm;
  /** The statistics collected on the Api level. */
  std::unique_ptr d_stats;
  /** The statistics registry (independent from any Solver's registry). */
  std::unique_ptr d_statsReg;
};

/* -------------------------------------------------------------------------- */
/* Solver                                                                     */
/* -------------------------------------------------------------------------- */

/**
 * A cvc5 solver.
 */
class CVC5_EXPORT Solver
{
  friend class Datatype;
  friend class DatatypeDecl;
  friend class DatatypeConstructor;
  friend class DatatypeConstructorDecl;
  friend class DatatypeSelector;
  friend class DriverOptions;
  friend class Grammar;
  friend class Plugin;
  friend class Op;
  friend class parser::Cmd;
  friend class Proof;
  friend class main::CommandExecutor;
  friend class Sort;
  friend class Term;

 public:
  /* .................................................................... */
  /* Constructors/Destructors                                             */
  /* .................................................................... */

  /**
   * Constructor.
   * @return The Solver.
   * @warning This constructor is deprecated and replaced by
   *          `Solver::Solver(TermManager&)`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use Solver::Solver(TermManager&) instead")]] Solver();
  /**
   * Constructor.
   *
   * Constructs solver instance from a given term manager instance.
   * @param tm The associated term manager.
   * @return The Solver.
   */
  Solver(TermManager& tm);

  /**
   * Destructor.
   */
  ~Solver();

  /**
   * Disallow copy/assignment.
   */
  Solver(const Solver&) = delete;
  Solver& operator=(const Solver&) = delete;

  /* .................................................................... */
  /* Sorts Handling                                                       */
  /* .................................................................... */

  /**
   * Get the Boolean sort.
   * @return Sort Boolean.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getBooleanSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::getBooleanSort() instead")]] Sort
  getBooleanSort() const;

  /**
   * Get the Integer sort.
   * @return Sort Integer.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getIntegerSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::getInteger() instead")]] Sort getIntegerSort()
      const;

  /**
   * Get the Real sort.
   * @return Sort Real.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getRealSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::getRealSort() instead")]] Sort getRealSort()
      const;

  /**
   * Get the regular expression sort.
   * @return Sort RegExp.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getRegExpSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::getRegExpSort() instead")]] Sort
  getRegExpSort() const;

  /**
   * Get the rounding mode sort.
   * @return Sort RoundingMode.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getRoundingModeSort()`. It will be removed in a
   * future release.
   */
  [[deprecated("Use TermManager::getRoundingModeSort() instead")]] Sort
  getRoundingModeSort() const;

  /**
   * Get the string sort.
   * @return Sort String.
   * @warning This function is deprecated and replaced by
   *          `TermManager::getStringSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::getStringSort() instead")]] Sort
  getStringSort() const;

  /**
   * Create an array sort.
   * @param indexSort The array index sort.
   * @param elemSort The array element sort.
   * @return The array sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkArraySort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkArraySort() instead")]] Sort mkArraySort(
      const Sort& indexSort, const Sort& elemSort) const;

  /**
   * Create a bit-vector sort.
   * @param size The bit-width of the bit-vector sort.
   * @return The bit-vector sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkBitVectorSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkBitVectorSort() instead")]] Sort
  mkBitVectorSort(uint32_t size) const;

  /**
   * Create a floating-point sort.
   * @param exp The bit-width of the exponent of the floating-point sort.
   * @param sig The bit-width of the significand of the floating-point sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointSort()`. It will be removed in a
   * future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointSort() instead")]] Sort
  mkFloatingPointSort(uint32_t exp, uint32_t sig) const;

  /**
   * Create a finite-field sort from a given string of
   * base n.
   *
   * @param size The modulus of the field. Must be prime.
   * @param base The base of the string representation of `size`.
   * @return The finite-field sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFiniteFieldSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkFiniteFieldSort() instead")]] Sort
  mkFiniteFieldSort(const std::string& size, uint32_t base = 10) const;

  /**
   * Create a datatype sort.
   * @param dtypedecl The datatype declaration from which the sort is created.
   * @return The datatype sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkDatatypeSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkDatatypeSort() instead")]] Sort
  mkDatatypeSort(const DatatypeDecl& dtypedecl) const;

  /**
   * Create a vector of datatype sorts.
   * @note The names of the datatype declarations must be distinct.
   * @param dtypedecls The datatype declarations from which the sort is created.
   * @return The datatype sorts.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkDatatypeSorts()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkDatatypeSorts() instead")]] std::vector
  mkDatatypeSorts(const std::vector& dtypedecls) const;

  /**
   * Create function sort.
   * @param sorts The sort of the function arguments.
   * @param codomain The sort of the function return value.
   * @return The function sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFunctionSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkFunctionSort() instead")]] Sort
  mkFunctionSort(const std::vector& sorts, const Sort& codomain) const;

  /**
   * Create a sort parameter.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param symbol The name of the sort.
   * @return The sort parameter.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkParamSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkParamSort() instead")]] Sort mkParamSort(
      const std::optional& symbol = std::nullopt) const;

  /**
   * Create a predicate sort.
   *
   * This is equivalent to calling mkFunctionSort() with the Boolean sort as the
   * codomain.
   * @param sorts The list of sorts of the predicate.
   * @return The predicate sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkPredicateSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkParamSort() instead")]] Sort mkPredicateSort(
      const std::vector& sorts) const;

  /**
   * Create a record sort
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param fields The list of fields of the record.
   * @return The record sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkRecordSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkRecordSort() instead")]] Sort mkRecordSort(
      const std::vector>& fields) const;

  /**
   * Create a set sort.
   * @param elemSort The sort of the set elements.
   * @return The set sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkSetSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkSetSort() instead")]] Sort mkSetSort(
      const Sort& elemSort) const;

  /**
   * Create a bag sort.
   * @param elemSort The sort of the bag elements.
   * @return The bag sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkBagSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkBagSort() instead")]] Sort mkBagSort(
      const Sort& elemSort) const;

  /**
   * Create a sequence sort.
   * @param elemSort The sort of the sequence elements.
   * @return The sequence sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkSequenceSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkSequenceSort() instead")]] Sort
  mkSequenceSort(const Sort& elemSort) const;

  /**
   * Create an abstract sort. An abstract sort represents a sort for a given
   * kind whose parameters and arguments are unspecified.
   *
   * The kind `k` must be the kind of a sort that can be abstracted, i.e., a
   * sort that has indices or argument sorts. For example, #ARRAY_SORT and
   * #BITVECTOR_SORT can be passed as the kind `k` to this function, while
   * #INTEGER_SORT and #STRING_SORT cannot.
   *
   * @note Providing the kind #ABSTRACT_SORT as an argument to this function
   * returns the (fully) unspecified sort, denoted `?`.
   *
   * @note Providing a kind `k` that has no indices and a fixed arity
   * of argument sorts will return the sort of kind `k` whose arguments are the
   * unspecified sort. For example, `mkAbstractSort(SortKind::ARRAY_SORT)` will
   * return the sort `(ARRAY_SORT ? ?)` instead of the abstract sort whose
   * abstract kind is #ARRAY_SORT.
   *
   * @param k The kind of the abstract sort
   * @return The abstract sort.
   *
   * @warning This function is experimental and may change in future versions.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkAbstractSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkAbstractSort() instead")]] Sort
  mkAbstractSort(SortKind k) const;

  /**
   * Create an uninterpreted sort.
   * @param symbol The name of the sort.
   * @return The uninterpreted sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkUninterpretedSort()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkUninterpretedSort() instead")]] Sort
  mkUninterpretedSort(
      const std::optional& symbol = std::nullopt) const;

  /**
   * Create an unresolved datatype sort.
   *
   * This is for creating yet unresolved sort placeholders for mutually
   * recursive parametric datatypes.
   *
   * @param symbol The symbol of the sort.
   * @param arity The number of sort parameters of the sort.
   * @return The unresolved sort.
   *
   * @warning This function is experimental and may change in future versions.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkUnresolvedDatatypeSort()`. It will be removed in
   *          a future release.
   */
  [[deprecated("Use TermManager::mkUnresolvedDatatypeSort() instead")]] Sort
  mkUnresolvedDatatypeSort(const std::string& symbol, size_t arity = 0) const;

  /**
   * Create an uninterpreted sort constructor sort.
   *
   * An uninterpreted sort constructor is an uninterpreted sort with arity > 0.
   *
   * @param symbol The symbol of the sort.
   * @param arity The arity of the sort (must be > 0)
   * @return The uninterpreted sort constructor sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkUninterpretedSortConstructorerSort()`. It will be
   *          removed in a future release.
   */
  [[deprecated(
      "Use TermManager::mkUninterpretedConstructorSort() instead")]] Sort
  mkUninterpretedSortConstructorSort(
      size_t arity,
      const std::optional& symbol = std::nullopt) const;

  /**
   * Create a tuple sort.
   * @param sorts The sorts of the elements of the tuple.
   * @return The tuple sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkTupleSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkTupleSort() instead")]] Sort mkTupleSort(
      const std::vector& sorts) const;

  /**
   * Create a nullable sort.
   * @param sort The sort of the element of the nullable.
   * @return The nullable sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableSort()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableSort() instead")]] Sort
  mkNullableSort(const Sort& sort) const;

  /* .................................................................... */
  /* Create Terms                                                         */
  /* .................................................................... */

  /**
   * Create n-ary term of given kind.
   * @param kind The kind of the term.
   * @param children The children of the term.
   * @return The Term
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkTerm()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkTerm() instead")]] Term mkTerm(
      Kind kind, const std::vector& children = {}) const;

  /**
   * Create n-ary term of given kind from a given operator.
   * Create operators with mkOp().
   * @param op The operator.
   * @param children The children of the term.
   * @return The Term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkTerm()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkTerm() instead")]] Term mkTerm(
      const Op& op, const std::vector& children = {}) const;

  /**
   * Create a tuple term.
   * @param terms The elements in the tuple.
   * @return The tuple Term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkTuple()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkTuple() instead")]] Term mkTuple(
      const std::vector& terms) const;
  /**
   * Create a nullable some term.
   * @param term The element value.
   * @return the Element value wrapped in some constructor.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableSome()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableSome() instead")]] Term
  mkNullableSome(const Term& term) const;
  /**
   * Create a selector for nullable term.
   * @param term A nullable term.
   * @return The element value of the nullable term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableVal()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableVal() instead")]] Term mkNullableVal(
      const Term& term) const;
  /**
   * Create a null tester for a nullable term.
   * @param term A nullable term.
   * @return A tester whether term is null.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableisNull()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableisNull() instead")]] Term
  mkNullableIsNull(const Term& term) const;
  /**
   * Create a some tester for a nullable term.
   * @param term A nullable term.
   * @return A tester whether term is some.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableisSome()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableisSome() instead")]] Term
  mkNullableIsSome(const Term& term) const;

  /**
   * Create a constant representing an null of the given sort.
   * @param sort The sort of the Nullable element.
   * @return The null constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableNull()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableNull() instead")]] Term
  mkNullableNull(const Sort& sort) const;
  /**
   * Create a term that lifts kind to nullable terms.
   * Example:
   * If we have the term ((_ nullable.lift +) x y),
   * where x, y of type (Nullable Int), then
   * kind would be ADD, and args would be [x, y].
   * This function would return
   * (nullable.lift (lambda ((a Int) (b Int)) (+ a b)) x y)
   * @param kind The lifted operator.
   * @param args The arguments of the lifted operator.
   * @return A term of Kind NULLABLE_LIFT where the first child
   * is a lambda expression, and the remaining children are
   * the original arguments.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkNullableLift()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkNullableLift() instead")]] Term
  mkNullableLift(Kind kind, const std::vector& args) const;

  /* .................................................................... */
  /* Create Operators                                                     */
  /* .................................................................... */

  /**
   * Create operator of Kind:
   *   - #BITVECTOR_EXTRACT
   *   - #BITVECTOR_REPEAT
   *   - #BITVECTOR_ROTATE_LEFT
   *   - #BITVECTOR_ROTATE_RIGHT
   *   - #BITVECTOR_SIGN_EXTEND
   *   - #BITVECTOR_ZERO_EXTEND
   *   - #DIVISIBLE
   *   - #FLOATINGPOINT_TO_FP_FROM_FP
   *   - #FLOATINGPOINT_TO_FP_FROM_IEEE_BV
   *   - #FLOATINGPOINT_TO_FP_FROM_REAL
   *   - #FLOATINGPOINT_TO_FP_FROM_SBV
   *   - #FLOATINGPOINT_TO_FP_FROM_UBV
   *   - #FLOATINGPOINT_TO_SBV
   *   - #FLOATINGPOINT_TO_UBV
   *   - #INT_TO_BITVECTOR
   *   - #TUPLE_PROJECT
   *
   * See cvc5::Kind for a description of the parameters.
   * @param kind The kind of the operator.
   * @param args The arguments (indices) of the operator.
   *
   * @note If ``args`` is empty, the Op simply wraps the cvc5::Kind.  The
   * Kind can be used in Solver::mkTerm directly without creating an Op
   * first.
   * @warning This function is deprecated and replaced by `TermManager::mkOp()`.
   *          It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkOp() instead")]] Op mkOp(
      Kind kind, const std::vector& args = {}) const;

#ifndef DOXYGEN_SKIP
  // Overload is only used to disambiguate the std::vector and std::string
  // overloads.
  [[deprecated("Use TermManager::mkOp() instead")]] Op mkOp(
      Kind kind, const std::initializer_list& args) const;
#endif

  /**
   * Create operator of kind:
   *   - #DIVISIBLE (to support arbitrary precision integers)
   * See cvc5::Kind for a description of the parameters.
   * @param kind The kind of the operator.
   * @param arg The string argument to this operator.
   * @warning This function is deprecated and replaced by `TermManager::mkOp()`.
   *          It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkOp() instead")]] Op mkOp(
      Kind kind, const std::string& arg) const;

  /* .................................................................... */
  /* Create Constants                                                     */
  /* .................................................................... */

  /**
   * Create a Boolean true constant.
   * @return The true constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkTrue()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkTrue() instead")]] Term mkTrue() const;

  /**
   * Create a Boolean false constant.
   * @return The false constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFalse()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkFalse() instead")]] Term mkFalse() const;

  /**
   * Create a Boolean constant.
   * @return The Boolean constant.
   * @param val The value of the constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkBoolean()`. It will be removed in a future
   * release.
   */
  [[deprecated("Use TermManager::mkBoolean() instead")]] Term mkBoolean(
      bool val) const;

  /**
   * Create a constant representing the number Pi.
   * @return A constant representing Pi.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkPi()`. It will be removed in a future release.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkPi()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkPi() instead")]] Term mkPi() const;
  /**
   * Create an integer constant from a string.
   * @param s The string representation of the constant, may represent an
   *          integer (e.g., "123").
   * @return A constant of sort Integer assuming `s` represents an integer)
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkInteger()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkInteger() instead")]] Term mkInteger(
      const std::string& s) const;

  /**
   * Create an integer constant from a c++ int.
   * @param val The value of the constant.
   * @return A constant of sort Integer.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkInteger()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkInteger() instead")]] Term mkInteger(
      int64_t val) const;

  /**
   * Create a real constant from a string.
   * @param s The string representation of the constant, may represent an
   *          integer (e.g., "123") or real constant (e.g., "12.34" or "12/34").
   * @return A constant of sort Real.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkReal()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkReal() instead")]] Term mkReal(
      const std::string& s) const;

  /**
   * Create a real constant from an integer.
   * @param val The value of the constant.
   * @return A constant of sort Real.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkReal()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkReal() instead")]] Term mkReal(
      int64_t val) const;

  /**
   * Create a real constant from a rational.
   * @param num The value of the numerator.
   * @param den The value of the denominator.
   * @return A constant of sort Real.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkReal()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkReal() instead")]] Term mkReal(
      int64_t num, int64_t den) const;

  /**
   * Create a regular expression all (re.all) term.
   * @return The all term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkRegExpAll()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkRegExpAll() instead")]] Term mkRegexpAll()
      const;

  /**
   * Create a regular expression allchar (re.allchar) term.
   * @return The allchar term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkRegExpAllChar()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkRegExpAllChar() instead")]] Term
  mkRegexpAllchar() const;

  /**
   * Create a regular expression none (re.none) term.
   * @return The none term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkRegExpNone()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkRegExpNone() instead")]] Term mkRegexpNone()
      const;

  /**
   * Create a constant representing an empty set of the given sort.
   * @param sort The sort of the set elements.
   * @return The empty set constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkEmptySet()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkEmptySet() instead")]] Term mkEmptySet(
      const Sort& sort) const;

  /**
   * Create a constant representing an empty bag of the given sort.
   * @param sort The sort of the bag elements.
   * @return The empty bag constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkEmptyBag()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkEmptyBag() instead")]] Term mkEmptyBag(
      const Sort& sort) const;

  /**
   * Create a separation logic empty term.
   * @return The separation logic empty term.
   * @warning This function is experimental and may change in future versions.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkSepEmp()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkSepEmp() instead")]] Term mkSepEmp() const;

  /**
   * Create a separation logic nil term.
   * @param sort The sort of the nil term.
   * @return The separation logic nil term.
   * @warning This function is experimental and may change in future versions.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkSepNil()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkSepNil() instead")]] Term mkSepNil(
      const Sort& sort) const;

  /**
   * Create a String constant from a `std::string` which may contain SMT-LIB
   * compatible escape sequences like `\u1234` to encode unicode characters.
   * @param s The string this constant represents.
   * @param useEscSequences Determines whether escape sequences in `s` should.
   * be converted to the corresponding unicode character
   * @return The String constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkString()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkString() instead")]] Term mkString(
      const std::string& s, bool useEscSequences = false) const;

  /**
   * Create a String constant from a `std::wstring`.
   * This function does not support escape sequences as `std::wstring` already
   * supports unicode characters.
   * @param s The string this constant represents.
   * @return The String constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkString()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkString() instead")]] Term mkString(
      const std::wstring& s) const;

  /**
   * Create an empty sequence of the given element sort.
   * @param sort The element sort of the sequence.
   * @return The empty sequence with given element sort.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkEmptySequence()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkEmptySequence() instead")]] Term
  mkEmptySequence(const Sort& sort) const;

  /**
   * Create a universe set of the given sort.
   * @param sort The sort of the set elements.
   * @return The universe set constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkUniverseSet()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkUniverseSet() instead")]] Term mkUniverseSet(
      const Sort& sort) const;

  /**
   * Create a bit-vector constant of given size and value.
   *
   * @note The given value must fit into a bit-vector of the given size.
   *
   * @param size The bit-width of the bit-vector sort.
   * @param val The value of the constant.
   * @return The bit-vector constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkBitVector()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkBitVector() instead")]] Term mkBitVector(
      uint32_t size, uint64_t val = 0) const;

  /**
   * Create a bit-vector constant of a given bit-width from a given string of
   * base 2, 10 or 16.
   *
   * @note The given value must fit into a bit-vector of the given size.
   *
   * @param size The bit-width of the constant.
   * @param s The string representation of the constant.
   * @param base The base of the string representation (`2` for binary, `10`
   * for decimal, and `16` for hexadecimal).
   * @return The bit-vector constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkBitVector()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkBitVector() instead")]] Term mkBitVector(
      uint32_t size, const std::string& s, uint32_t base) const;

  /**
   * Create a finite field constant in a given field from a given string
   * of base n.
   *
   * @param value The string representation of the constant.
   * @param sort  The field sort.
   * @param base  The base of the string representation of `value`.
   *
   * If `size` is the field size, the constant needs not be in the range
   * [0,size). If it is outside this range, it will be reduced modulo size
   * before being constructed.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFiniteFieldElem()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkFiniteFieldElem() instead")]] Term
  mkFiniteFieldElem(const std::string& value,
                    const Sort& sort,
                    uint32_t base = 10) const;

  /**
   * Create a constant array with the provided constant value stored at every
   * index.
   * @param sort The sort of the constant array (must be an array sort).
   * @param val The constant value to store (must match the sort's element
   *            sort).
   * @return The constant array term.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkConstArray()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkConstArray() instead")]] Term mkConstArray(
      const Sort& sort, const Term& val) const;

  /**
   * Create a positive infinity floating-point constant (SMT-LIB: `+oo`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointPosInf()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointPosInf() instead")]] Term
  mkFloatingPointPosInf(uint32_t exp, uint32_t sig) const;

  /**
   * Create a negative infinity floating-point constant (SMT-LIB: `-oo`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointNegInf()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointNegInf() instead")]] Term
  mkFloatingPointNegInf(uint32_t exp, uint32_t sig) const;

  /**
   * Create a not-a-number floating-point constant (SMT-LIB: `NaN`).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointNaN()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointNaN() instead")]] Term
  mkFloatingPointNaN(uint32_t exp, uint32_t sig) const;

  /**
   * Create a positive zero floating-point constant (SMT-LIB: +zero).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointPosZero()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointPosZero() instead")]] Term
  mkFloatingPointPosZero(uint32_t exp, uint32_t sig) const;

  /**
   * Create a negative zero floating-point constant (SMT-LIB: -zero).
   * @param exp Number of bits in the exponent.
   * @param sig Number of bits in the significand.
   * @return The floating-point constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPointNegZero()`. It will be removed in a
   *          future release.
   */
  [[deprecated("Use TermManager::mkFloatingPointNegZero() instead")]] Term
  mkFloatingPointNegZero(uint32_t exp, uint32_t sig) const;

  /**
   * Create a rounding mode value.
   * @param rm The floating point rounding mode this constant represents.
   * @return The rounding mode value.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkRoundingMode()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkRoundingMode() instead")]] Term
  mkRoundingMode(RoundingMode rm) const;

  /**
   * Create a floating-point value from a bit-vector given in IEEE-754
   * format.
   * @param exp Size of the exponent.
   * @param sig Size of the significand.
   * @param val Value of the floating-point constant as a bit-vector term.
   * @return The floating-point value.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPoint()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkFloatingPoint() instead")]] Term
  mkFloatingPoint(uint32_t exp, uint32_t sig, const Term& val) const;
  /**
   * Create a floating-point value from its three IEEE-754 bit-vector
   * value components (sign bit, exponent, significand).
   * @param sign The sign bit.
   * @param exp  The bit-vector representing the exponent.
   * @param sig The bit-vector representing the significand.
   * @return The floating-point value.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkFloatingPoint()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkFloatingPoint() instead")]] Term
  mkFloatingPoint(const Term& sign, const Term& exp, const Term& sig) const;

  /**
   * Create a cardinality constraint for an uninterpreted sort.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param sort The sort the cardinality constraint is for.
   * @param upperBound The upper bound on the cardinality of the sort.
   * @return The cardinality constraint.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkCardinalityConstraint()`. It will be removed in a
   * future release.
   */
  [[deprecated("Use TermManager::mkCardinalityConstraint() instead")]] Term
  mkCardinalityConstraint(const Sort& sort, uint32_t upperBound) const;

  /* .................................................................... */
  /* Create Variables                                                     */
  /* .................................................................... */

  /**
   * Create a free constant.
   *
   * Note that the returned term is always fresh, even if the same arguments
   * were provided on a previous call to mkConst().
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-const  )
   *     (declare-fun  () )
   * \endverbatim
   *
   * @param sort The sort of the constant.
   * @param symbol The name of the constant (optional).
   * @return The constant.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkConst()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkConst() instead")]] Term mkConst(
      const Sort& sort,
      const std::optional& symbol = std::nullopt) const;

  /**
   * Create a bound variable to be used in a binder (i.e., a quantifier, a
   * lambda, or a witness binder).
   *
   * Note that the returned term is always fresh, even if the same arguments
   * were provided on a previous call to mkConst.
   *
   * @param sort The sort of the variable.
   * @param symbol The name of the variable (optional).
   * @return The variable.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkVar()`. It will be removed in a future release.
   */
  [[deprecated("Use TermManager::mkVar() instead")]] Term mkVar(
      const Sort& sort,
      const std::optional& symbol = std::nullopt) const;

  /* .................................................................... */
  /* Create datatype constructor declarations                             */
  /* .................................................................... */

  /**
   * Create a datatype constructor declaration.
   * @param name The name of the datatype constructor.
   * @return The DatatypeConstructorDecl.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkDatatypeConstructorDecl()`. It will be removed in
   *          a future release.
   */
  [[deprecated(
      "Use TermManager::mkDatatypeConstructorDecl() "
      "instead")]] DatatypeConstructorDecl
  mkDatatypeConstructorDecl(const std::string& name);

  /* .................................................................... */
  /* Create datatype declarations                                         */
  /* .................................................................... */

  /**
   * Create a datatype declaration.
   * @param name The name of the datatype.
   * @param isCoDatatype True if a codatatype is to be constructed.
   * @return The DatatypeDecl.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkDatatypeDecl()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkDatatypeDecl() instead")]] DatatypeDecl
  mkDatatypeDecl(const std::string& name, bool isCoDatatype = false);

  /**
   * Create a datatype declaration.
   * Create sorts parameter with Solver::mkParamSort().
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param name The name of the datatype.
   * @param params A list of sort parameters.
   * @param isCoDatatype True if a codatatype is to be constructed.
   * @return The DatatypeDecl.
   * @warning This function is deprecated and replaced by
   *          `TermManager::mkDatatypeDecl()`. It will be removed in a future
   *          release.
   */
  [[deprecated("Use TermManager::mkDatatypeDecl() instead")]] DatatypeDecl
  mkDatatypeDecl(const std::string& name,
                 const std::vector& params,
                 bool isCoDatatype = false);

  /* .................................................................... */
  /* Formula Handling                                                     */
  /* .................................................................... */

  /**
   * Simplify a term or formula based on rewriting and (optionally) applying
   * substitutions for solved variables.
   *
   * If applySubs is true, then for example, if `(= x 0)` was asserted to this
   * solver, this method may replace occurrences of `x` with `0`.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param t         The term to simplify.
   * @param applySubs True to apply substitutions for solved variables.
   * @return The simplified term.
   */
  Term simplify(const Term& t, bool applySubs = false);

  /**
   * Assert a formula.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (assert )
   * \endverbatim
   *
   * @param term The formula to assert.
   */
  void assertFormula(const Term& term) const;

  /**
   * Check satisfiability.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (check-sat)
   * \endverbatim
   *
   * @return The result of the satisfiability check.
   */
  Result checkSat() const;

  /**
   * Check satisfiability assuming the given formula.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (check-sat-assuming (  ))
   * \endverbatim
   *
   * @param assumption The formula to assume.
   * @return The result of the satisfiability check.
   */
  Result checkSatAssuming(const Term& assumption) const;

  /**
   * Check satisfiability assuming the given formulas.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (check-sat-assuming ( + ))
   * \endverbatim
   *
   * @param assumptions The formulas to assume.
   * @return The result of the satisfiability check.
   */
  Result checkSatAssuming(const std::vector& assumptions) const;

  /**
   * Create datatype sort.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-datatype  )
   * \endverbatim
   *
   * @param symbol The name of the datatype sort.
   * @param ctors The constructor declarations of the datatype sort.
   * @return The datatype sort.
   */
  Sort declareDatatype(const std::string& symbol,
                       const std::vector& ctors) const;

  /**
   * Declare n-ary function symbol.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-fun  ( * ) )
   * \endverbatim
   *
   * @param symbol The name of the function.
   * @param sorts The sorts of the parameters to this function.
   * @param sort The sort of the return value of this function.
   * @param fresh If true, then this method always returns a new Term.
   * Otherwise, this method will always return the same Term
   * for each call with the given sorts and symbol where fresh is false.
   * @return The function.
   */
  Term declareFun(const std::string& symbol,
                  const std::vector& sorts,
                  const Sort& sort,
                  bool fresh = true) const;

  /**
   * Declare uninterpreted sort.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-sort  )
   * \endverbatim
   *
   * @note This corresponds to
   *       mkUninterpretedSort(const std::optional&)
   *       if arity = 0, and to
   *       mkUninterpretedSortConstructorSort(size_t arity, const
   * std::optional&) if arity > 0.
   *
   * @param symbol The name of the sort.
   * @param arity The arity of the sort.
   * @param fresh If true, then this method always returns a new Sort.
   * Otherwise, this method will always return the same Sort
   * for each call with the given arity and symbol where fresh is false.
   * @return The sort.
   */
  Sort declareSort(const std::string& symbol,
                   uint32_t arity,
                   bool fresh = true) const;

  /**
   * Define n-ary function.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (define-fun )
   * \endverbatim
   *
   * @param symbol The name of the function.
   * @param bound_vars The parameters to this function.
   * @param sort The sort of the return value of this function.
   * @param term The function body.
   * @param global Determines whether this definition is global (i.e., persists
   *               when popping the context).
   * @return The function.
   */
  Term defineFun(const std::string& symbol,
                 const std::vector& bound_vars,
                 const Sort& sort,
                 const Term& term,
                 bool global = false) const;

  /**
   * Define recursive function.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (define-fun-rec )
   * \endverbatim
   *
   * @param symbol The name of the function.
   * @param bound_vars The parameters to this function.
   * @param sort The sort of the return value of this function.
   * @param term The function body.
   * @param global Determines whether this definition is global (i.e., persists
   *               when popping the context).
   * @return The function.
   */
  Term defineFunRec(const std::string& symbol,
                    const std::vector& bound_vars,
                    const Sort& sort,
                    const Term& term,
                    bool global = false) const;

  /**
   * Define recursive function.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (define-fun-rec )
   * \endverbatim
   *
   * Create parameter `fun` with TermManager::mkConst().
   *
   * @param fun The sorted function.
   * @param bound_vars The parameters to this function.
   * @param term The function body.
   * @param global Determines whether this definition is global (i.e., persists
   *               when popping the context).
   * @return The function.
   */
  Term defineFunRec(const Term& fun,
                    const std::vector& bound_vars,
                    const Term& term,
                    bool global = false) const;

  /**
   * Define recursive functions.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (define-funs-rec
   *         ( _1 ... _n )
   *         ( _1 ... _n )
   *     )
   * \endverbatim
   *
   * Create elements of parameter `funs` with `TermManager::mkConst()`.
   *
   * @param funs The sorted functions.
   * @param bound_vars The list of parameters to the functions.
   * @param terms The list of function bodies of the functions.
   * @param global Determines whether this definition is global (i.e., persists
   *               when popping the context).
   */
  void defineFunsRec(const std::vector& funs,
                     const std::vector>& bound_vars,
                     const std::vector& terms,
                     bool global = false) const;

  /**
   * Get the list of asserted formulas.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-assertions)
   * \endverbatim
   *
   * @return The list of asserted formulas.
   */
  std::vector getAssertions() const;

  /**
   * Get info from the solver.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-info )
   * \endverbatim
   *
   * @return The info.
   */
  std::string getInfo(const std::string& flag) const;

  /**
   * Get the value of a given option.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-option )
   * \endverbatim
   *
   * @param option The option for which the value is queried.
   * @return A string representation of the option value.
   */
  std::string getOption(const std::string& option) const;

  /**
   * Get all option names that can be used with setOption(), getOption() and
   * getOptionInfo().
   * @return All option names.
   */
  std::vector getOptionNames() const;

  /**
   * Get some information about the given option.
   *
   * Check the OptionInfo class for more details on which information is
   * available.
   *
   * @return Information about the given option.
   */
  OptionInfo getOptionInfo(const std::string& option) const;

  /**
   * Get the driver options, which provide access to options that can not be
   * communicated properly via getOption() and getOptionInfo().
   * @return A DriverOptions object.
   */
  DriverOptions getDriverOptions() const;

  /**
   * Get the set of unsat ("failed") assumptions.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-unsat-assumptions)
   *
   * Requires to enable option
   * :ref:`produce-unsat-assumptions `.
   * \endverbatim
   *
   * @return The set of unsat assumptions.
   */
  std::vector getUnsatAssumptions() const;

  /**
   * Get the unsatisfiable core.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-unsat-core)
   *
   * Requires to enable option
   * :ref:`produce-unsat-cores `.
   *
   * .. note::
   *   In contrast to SMT-LIB, cvc5's API does not distinguish between named
   *   and unnamed assertions when producing an unsatisfiable core.
   *   Additionally, the API allows this option to be called after a check with
   *   assumptions. A subset of those assumptions may be included in the
   *   unsatisfiable core returned by this function.
   * \endverbatim
   *
   * @return A set of terms representing the unsatisfiable core.
   */
  std::vector getUnsatCore() const;

  /**
   * Get the lemmas used to derive unsatisfiability.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-unsat-core-lemmas)
   *
   * Requires the SAT proof unsat core mode, so to enable option
   * :ref:`unsat-cores-mode=sat-proof `.
   *
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return A set of terms representing the lemmas used to derive
   * unsatisfiability.
   */
  std::vector getUnsatCoreLemmas() const;

  /**
   * Get a difficulty estimate for an asserted formula. This function is
   * intended to be called immediately after any response to a checkSat.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return A map from (a subset of) the input assertions to a real value that.
   *         is an estimate of how difficult each assertion was to solve.
   *         Unmentioned assertions can be assumed to have zero difficulty.
   */
  std::map getDifficulty() const;

  /**
   * Get a timeout core.
   *
   * \verbatim embed:rst:leading-asterisk
   * This function computes a subset of the current assertions that cause a
   * timeout. It may make multiple checks for satisfiability internally, each
   * limited by the timeout value given by
   * :ref:`timeout-core-timeout `.
   *
   * If the result is unknown and the reason is timeout, then returned the set
   * of assertions corresponds to a subset of the current assertions that cause
   * a timeout in the specified time :ref:`timeout-core-timeout
   * `. If the result is unsat, then the list
   * of formulas correspond to an unsat core for the current assertions.
   * Otherwise, the result is sat, indicating that the current assertions are
   * satisfiable, and the returned set of assertions is empty.
   * \endverbatim
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-timeout-core)
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return The result of the timeout core computation. This is a pair
   *         containing a result and a set of assertions.
   */
  std::pair> getTimeoutCore() const;

  /**
   * Get a timeout core of the given assumptions.
   *
   * This function computes a subset of the given assumptions that cause a
   * timeout when added to the current assertions.
   *
   * \verbatim embed:rst:leading-asterisk
   * If the result is unknown and the reason is timeout, then the set of
   * assumptions corresponds to a subset of the given assumptions that cause a
   * timeout when added to the current assertions in the specified time
   * :ref:`timeout-core-timeout `. If the
   * result is unsat, then the set of assumptions together with the current
   * assertions correspond to an unsat core for the current assertions.
   * Otherwise, the result is sat, indicating that the given assumptions plus
   * the current assertions are satisfiable, and the returned set of
   * assumptions is empty.
   * \endverbatim
   *
   * @note This command does not require being preceeded by a call to
   *       `checkSat()`.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-timeout-core (*))
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param assumptions The (non-empty) set of formulas to assume.
   *
   * @return The result of the timeout core computation. This is a pair
   *         containing a result and a set of assumptions.
   */
  std::pair> getTimeoutCoreAssuming(
      const std::vector& assumptions) const;
  /**
   * Get a proof associated with the most recent call to checkSat.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-proof :c)
   *
   * Requires to enable option
   * :ref:`produce-proofs `.
   * The string representation depends on the value of option
   * :ref:`produce-proofs `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param c The component of the proof to return
   * @return A vector of proofs.
   */
  std::vector getProof(
      modes::ProofComponent c = modes::ProofComponent::FULL) const;

  /**
   * Prints a proof as a string in a selected proof format mode.
   * Other aspects of printing are taken from the solver options.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param proof A proof, usually obtained from Solver::getProof().
   * @param format The proof format used to print the proof.  Must be
   * `modes::ProofFormat::NONE` if the proof is from a component other than
   * `modes::ProofComponent::FULL`.
   * @param assertionNames Mapping between assertions and names, if they were
   * given by the user.
   * @return The string representation of the proof in the given format.
   */
  std::string proofToString(
      Proof proof,
      modes::ProofFormat format = modes::ProofFormat::DEFAULT,
      const std::map& assertionNames =
          std::map()) const;

  /**
   * Get a list of learned literals that are entailed by the current set of
   * assertions.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param t The type of learned literals to return
   * @return A list of literals that were learned at top-level.
   */
  std::vector getLearnedLiterals(
      modes::LearnedLitType t = modes::LearnedLitType::INPUT) const;

  /**
   * Get the value of the given term in the current model.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-value (  ))
   * \endverbatim
   *
   * @param term The term for which the value is queried.
   * @return The value of the given term.
   */
  Term getValue(const Term& term) const;

  /**
   * Get the values of the given terms in the current model.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-value ( * ))
   * \endverbatim
   *
   * @param terms The terms for which the value is queried.
   * @return The values of the given terms.
   */
  std::vector getValue(const std::vector& terms) const;

  /**
   * Get the domain elements of uninterpreted sort s in the current model. The
   * current model interprets s as the finite sort whose domain elements are
   * given in the return value of this function.
   *
   * @param s The uninterpreted sort in question.
   * @return The domain elements of s in the current model.
   */
  std::vector getModelDomainElements(const Sort& s) const;

  /**
   * Determine if the model value of the given free constant was essential for
   * showing satisfiability of the last `checkSat()` query based on the current
   * model.
   *
   * For any free constant `v`, this will only return false if
   * \verbatim embed:rst:inline :ref:`model-cores
   * `\endverbatim
   * has been set to true.
   * @warning This function is experimental and may change in future versions.
   *
   * @param v The term in question.
   * @return True if `v` is a model core symbol.
   */
  bool isModelCoreSymbol(const Term& v) const;

  /**
   * Get the model
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-model)
   *
   * Requires to enable option
   * :ref:`produce-models `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param sorts The list of uninterpreted sorts that should be printed in
   *              the model.
   * @param consts The list of free constants that should be printed in the
   *               model. A subset of these may be printed based on
   *               isModelCoreSymbol().
   * @return A string representing the model.
   */
  std::string getModel(const std::vector& sorts,
                       const std::vector& consts) const;

  /**
   * Do quantifier elimination.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-qe )
   * \endverbatim
   *
   * @note Quantifier Elimination is is only complete for logics such as LRA,
   * LIA and BV.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param q A quantified formula of the form
   *          @f$Q\bar{x}_1... Q\bar{x}_n. P( x_1...x_i, y_1...y_j)@f$
   *          where
   *          @f$Q\bar{x}@f$ is a set of quantified variables of the form
   *          @f$Q x_1...x_k@f$ and
   *          @f$P( x_1...x_i, y_1...y_j )@f$ is a quantifier-free formula
   * @return A formula @f$\phi@f$  such that, given the current set of formulas
   *         @f$A@f$ asserted to this solver:
   *         - @f$(A \wedge q)@f$ and @f$(A \wedge \phi)@f$ are equivalent
   *         - @f$\phi@f$ is quantifier-free formula containing only free
   *           variables in @f$y_1...y_n@f$.
   */
  Term getQuantifierElimination(const Term& q) const;

  /**
   * Do partial quantifier elimination, which can be used for incrementally
   * computing the result of a quantifier elimination.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-qe-disjunct )
   * \endverbatim
   *
   * @note Quantifier Elimination is is only complete for logics such as LRA,
   * LIA and BV.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param q A quantified formula of the form
   *          @f$Q\bar{x}_1... Q\bar{x}_n. P( x_1...x_i, y_1...y_j)@f$
   *          where
   *          @f$Q\bar{x}@f$ is a set of quantified variables of the form
   *          @f$Q x_1...x_k@f$ and
   *          @f$P( x_1...x_i, y_1...y_j )@f$ is a quantifier-free formula
   * @return A formula @f$\phi@f$ such that, given the current set of formulas
   *         @f$A@f$ asserted to this solver:
   *         - @f$(A \wedge q \implies A \wedge \phi)@f$ if @f$Q@f$ is
   *           @f$\forall@f$, and @f$(A \wedge \phi \implies A \wedge q)@f$ if
   *           @f$Q@f$ is @f$\exists@f$
   *         - @f$\phi@f$ is quantifier-free formula containing only free
   *           variables in @f$y_1...y_n@f$
   *         - If @f$Q@f$ is @f$\exists@f$, let @f$(A \wedge Q_n)@f$ be the
   *           formula
   *           @f$(A \wedge \neg (\phi \wedge Q_1) \wedge ... \wedge
   *           \neg (\phi \wedge Q_n))@f$
   *           where for each @f$i = 1...n@f$,
   *           formula @f$(\phi \wedge Q_i)@f$ is the result of calling
   *           Solver::getQuantifierEliminationDisjunct() for @f$q@f$ with the
   *           set of assertions @f$(A \wedge Q_{i-1})@f$.
   *           Similarly, if @f$Q@f$ is @f$\forall@f$, then let
   *           @f$(A \wedge Q_n)@f$ be
   *           @f$(A \wedge (\phi \wedge Q_1) \wedge ... \wedge (\phi \wedge
   *           Q_n))@f$
   *           where @f$(\phi \wedge Q_i)@f$ is the same as above.
   *           In either case, we have that @f$(\phi \wedge Q_j)@f$ will
   *           eventually be true or false, for some finite j.
   */
  Term getQuantifierEliminationDisjunct(const Term& q) const;

  /**
   * When using separation logic, this sets the location sort and the
   * datatype sort to the given ones. This function should be invoked exactly
   * once, before any separation logic constraints are provided.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param locSort The location sort of the heap.
   * @param dataSort The data sort of the heap.
   */
  void declareSepHeap(const Sort& locSort, const Sort& dataSort) const;

  /**
   * When using separation logic, obtain the term for the heap.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return The term for the heap.
   */
  Term getValueSepHeap() const;

  /**
   * When using separation logic, obtain the term for nil.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return The term for nil.
   */
  Term getValueSepNil() const;

  /**
   * Declare a symbolic pool of terms with the given initial value.
   *
   * For details on how pools are used to specify instructions for quantifier
   * instantiation, see documentation for the #INST_POOL kind.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (declare-pool   ( * ))
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param symbol The name of the pool.
   * @param sort The sort of the elements of the pool.
   * @param initValue The initial value of the pool.
   * @return The pool symbol.
   */
  Term declarePool(const std::string& symbol,
                   const Sort& sort,
                   const std::vector& initValue) const;
  /**
   * Declare an oracle function with reference to an implementation.
   *
   * Oracle functions have a different semantics with respect to ordinary
   * declared functions. In particular, for an input to be satisfiable,
   * its oracle functions are implicitly universally quantified.
   *
   * This function is used in part for implementing this command:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   * (declare-oracle-fun  (*)  )
   * \endverbatim
   *
   * In particular, the above command is implemented by constructing a
   * function over terms that wraps a call to binary sym via a text interface.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param symbol The name of the oracle
   * @param sorts The sorts of the parameters to this function
   * @param sort The sort of the return value of this function
   * @param fn The function that implements the oracle function.
   * @return The oracle function
   */
  Term declareOracleFun(const std::string& symbol,
                        const std::vector& sorts,
                        const Sort& sort,
                        std::function&)> fn) const;
  /**
   * Add plugin to this solver. Its callbacks will be called throughout the
   * lifetime of this solver.
   * @warning This function is experimental and may change in future versions.
   * @param p The plugin to add to this solver.
   */
  void addPlugin(Plugin& p);
  /**
   * Pop (a) level(s) from the assertion stack.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (pop )
   * \endverbatim
   *
   * @param nscopes The number of levels to pop.
   */
  void pop(uint32_t nscopes = 1) const;

  /**
   * Get an interpolant
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-interpolant )
   *
   * Requires option
   * :ref:`produce-interpolants ` to be set to
   * a mode different from `none`. \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param conj The conjecture term.
   * @return A Term @f$I@f$ such that @f$A \rightarrow I@f$ and
   *         @f$I \rightarrow B@f$ are valid, where @f$A@f$ is the
   *         current set of assertions and @f$B@f$ is given in the input by
   *         `conj`, or the null term if such a term cannot be found.
   */
  Term getInterpolant(const Term& conj) const;

  /**
   * Get an interpolant
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-interpolant  )
   *
   * Requires option
   * :ref:`produce-interpolants ` to be set to
   * a mode different from `none`. \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param conj The conjecture term.
   * @param grammar The grammar for the interpolant I.
   * @return A Term @f$I@f$ such that @f$A \rightarrow I@f$ and
   *         @f$I \rightarrow B@f$ are valid, where @f$A@f$ is the
   *         current set of assertions and @f$B@f$ is given in the input by
   *         `conj`, or the null term if such a term cannot be found.
   */
  Term getInterpolant(const Term& conj, Grammar& grammar) const;

  /**
   * Get the next interpolant. Can only be called immediately after a successful
   * call to get-interpolant or get-interpolant-next. Is guaranteed to produce a
   * syntactically different interpolant wrt the last returned interpolant if
   * successful.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-interpolant-next)
   *
   * Requires to enable incremental mode, and option
   * :ref:`produce-interpolants ` to be set to
   * a mode different from `none`. \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return A Term @f$I@f$ such that @f$A \rightarrow I@f$ and
   *         @f$I \rightarrow B@f$ are valid, where @f$A@f$ is the
   *         current set of assertions and @f$B@f$ is given in the input by
   *         `conj`, or the null term if such a term cannot be found.
   */
  Term getInterpolantNext() const;

  /**
   * Get an abduct.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-abduct )
   *
   * Requires to enable option
   * :ref:`produce-abducts `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param conj The conjecture term.
   * @return A term @f$C@f$ such that @f$(A \wedge C)@f$ is satisfiable,
   *         and @f$(A \wedge \neg B \wedge C)@f$ is unsatisfiable, where
   *         @f$A@f$ is the current set of assertions and @f$B@f$ is
   *         given in the input by ``conj``, or the null term if such a term
   *         cannot be found.
   */
  Term getAbduct(const Term& conj) const;

  /**
   * Get an abduct.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-abduct  )
   *
   * Requires to enable option
   * :ref:`produce-abducts `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   *
   * @param conj The conjecture term.
   * @param grammar The grammar for the abduct @f$C@f$
   * @return A term C such that @f$(A \wedge C)@f$ is satisfiable, and
   *        @f$(A \wedge \neg B \wedge C)@f$ is unsatisfiable, where @f$A@f$ is
   *        the current set of assertions and @f$B@f$ is given in the input by
   *        `conj`, or the null term if such a term cannot be found.
   */
  Term getAbduct(const Term& conj, Grammar& grammar) const;

  /**
   * Get the next abduct. Can only be called immediately after a successful
   * call to get-abduct or get-abduct-next. Is guaranteed to produce a
   * syntactically different abduct wrt the last returned abduct if successful.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (get-abduct-next)
   *
   * Requires to enable incremental mode, and option
   * :ref:`produce-abducts `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @return A term C such that @f$(A \wedge C)@f$ is satisfiable, and
   *        @f$(A \wedge \neg B \wedge C)@f$ is unsatisfiable, where @f$A@f$ is
   *        the current set of assertions and @f$B@f$ is given in the input by
   *        the last call to getAbduct(), or the null term if such a term
   *        cannot be found.
   */
  Term getAbductNext() const;

  /**
   * Block the current model. Can be called only if immediately preceded by a
   * SAT or INVALID query.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (block-model)
   *
   * Requires enabling option
   * :ref:`produce-models `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param mode The mode to use for blocking.
   */
  void blockModel(modes::BlockModelsMode mode) const;

  /**
   * Block the current model values of (at least) the values in terms. Can be
   * called only if immediately preceded by a SAT query.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (block-model-values ( + ))
   *
   * Requires enabling option
   * :ref:`produce-models `.
   * \endverbatim
   *
   * @warning This function is experimental and may change in future versions.
   * @param terms The model values to block.
   */
  void blockModelValues(const std::vector& terms) const;

  /**
   * @warning This function is experimental and may change in future versions.
   *
   * @return A string that contains information about all instantiations made
   *         by the quantifiers module.
   */
  std::string getInstantiations() const;

  /**
   * Push (a) level(s) to the assertion stack.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (push )
   * \endverbatim
   *
   * @param nscopes The number of levels to push.
   */
  void push(uint32_t nscopes = 1) const;

  /**
   * Remove all assertions.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (reset-assertions)
   * \endverbatim
   *
   */
  void resetAssertions() const;

  /**
   * Set info.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (set-info )
   * \endverbatim
   *
   * @param keyword The info flag.
   * @param value The value of the info flag.
   */
  void setInfo(const std::string& keyword, const std::string& value) const;

  /**
   * Set logic.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (set-logic )
   * \endverbatim
   *
   * @param logic The logic to set.
   */
  void setLogic(const std::string& logic) const;

  /**
   * Determine if `setLogic()` has been called.
   *
   * @return True if `setLogic()` has already been called for this solver
   *         instance.
   */
  bool isLogicSet() const;

  /**
   * Get the logic set the solver.
   *
   * @note Asserts isLogicSet().
   *
   * @return The logic used by the solver.
   */
  std::string getLogic() const;

  /**
   * Set option.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (set-option :
  )
   * \endverbatim
   *
   * @param inv The function-to-synthesize.
   * @param pre The pre-condition.
   * @param trans The transition relation.
   * @param post The post-condition.
   */
  void addSygusInvConstraint(const Term& inv,
                             const Term& pre,
                             const Term& trans,
                             const Term& post) const;

  /**
   * Try to find a solution for the synthesis conjecture corresponding to the
   * current list of functions-to-synthesize, universal variables and
   * constraints.
   *
   * SyGuS v2:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (check-synth)
   * \endverbatim
   *
   * @return The result of the check, which is "solution" if the check found a
   *         solution in which case solutions are available via
   *         getSynthSolutions, "no solution" if it was determined there is no
   *         solution, or "unknown" otherwise.
   */
  SynthResult checkSynth() const;

  /**
   * Try to find a next solution for the synthesis conjecture corresponding to
   * the current list of functions-to-synthesize, universal variables and
   * constraints. Must be called immediately after a successful call to
   * check-synth or check-synth-next. Requires incremental mode.
   *
   * SyGuS v2:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (check-synth-next)
   * \endverbatim
   *
   * @return The result of the check, which is "solution" if the check found a
   *         solution in which case solutions are available via
   *         getSynthSolutions, "no solution" if it was determined there is no
   *         solution, or "unknown" otherwise.
   */
  SynthResult checkSynthNext() const;

  /**
   * Get the synthesis solution of the given term. This function should be
   * called immediately after the solver answers unsat for sygus input.
   * @param term The term for which the synthesis solution is queried.
   * @return The synthesis solution of the given term.
   */
  Term getSynthSolution(const Term& term) const;

  /**
   * Get the synthesis solutions of the given terms. This function should be
   * called immediately after the solver answers unsat for sygus input.
   * @param terms The terms for which the synthesis solutions is queried.
   * @return The synthesis solutions of the given terms.
   */
  std::vector getSynthSolutions(const std::vector& terms) const;

  /**
   * Find a target term of interest using sygus enumeration, with no provided
   * grammar.
   *
   * The solver will infer which grammar to use in this call, which by default
   * will be the grammars specified by the function(s)-to-synthesize in the
   * current context.
   *
   * SyGuS v2:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (find-synth :target)
   * \endverbatim
   *
   * @param fst The identifier specifying what kind of term to find
   * @return The result of the find, which is the null term if this call failed.
   *
   * @warning This function is experimental and may change in future versions.
   */
  Term findSynth(modes::FindSynthTarget fst) const;
  /**
   * Find a target term of interest using sygus enumeration with a provided
   * grammar.
   *
   * SyGuS v2:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (find-synth :target G)
   * \endverbatim
   *
   * @param fst The identifier specifying what kind of term to find
   * @param grammar The grammar for the term
   * @return The result of the find, which is the null term if this call failed.
   *
   * @warning This function is experimental and may change in future versions.
   */
  Term findSynth(modes::FindSynthTarget fst, Grammar& grammar) const;
  /**
   * Try to find a next target term of interest using sygus enumeration. Must
   * be called immediately after a successful call to find-synth or
   * find-synth-next.
   *
   * SyGuS v2:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (find-synth-next)
   * \endverbatim
   *
   * @return The result of the find, which is the null term if this call failed.
   *
   * @warning This function is experimental and may change in future versions.
   */
  Term findSynthNext() const;

  /**
   * Get a snapshot of the current state of the statistic values of this
   * solver. The returned object is completely decoupled from the solver and
   * will not change when the solver is used again.
   * @return A snapshot of the current state of the statistic values.
   */
  Statistics getStatistics() const;

  /**
   * Print the statistics to the given file descriptor, suitable for usage in
   * signal handlers.
   * @param fd The file descriptor.
   */
  void printStatisticsSafe(int fd) const;

  /**
   * Determines if the output stream for the given tag is enabled. Tags can be
   * enabled with the `output` option (and `-o ` on the command line).
   *
   * Requires that a valid tag is given.
   *
   * @return True if the given tag is enabled.
   */
  bool isOutputOn(const std::string& tag) const;

  /**
   * Get an output stream for the given tag.
   *
   * Tags can be enabled with the `output` option (and `-o ` on the
   * command line). Raises an exception when an invalid tag is given.
   *
   * @warning This function is experimental and may change in future versions.
   *
   * @param tag The output tag.
   * @return The output stream.
   */
  std::ostream& getOutput(const std::string& tag) const;

  /**
   * Get a string representation of the version of this solver.
   * @return The version string.
   */
  std::string getVersion() const;

  /**
   * Get the associated term manager instance.
   * @return The term manager.
   */
  TermManager& getTermManager() const;

 private:
  /**
   * Constructs a solver with the given original options. This should only be
   * used internally when the Solver is reset.
   * @param tm       The associated term manager.
   * @param original The original set of configuration options.
   */
  Solver(TermManager& tm, std::unique_ptr&& original);

  /**
   * Synthesize n-ary function following specified syntactic constraints.
   *
   * SMT-LIB:
   *
   * \verbatim embed:rst:leading-asterisk
   * .. code:: smtlib
   *
   *     (synth-fun  ( * )  ?)
   * \endverbatim
   *
   * @param symbol The name of the function.
   * @param boundVars The parameters to this function.
   * @param sort The sort of the return value of this function.
   * @param isInv Determines whether this is `synth-fun` or `synth-inv`.
   * @param grammar The syntactic constraints.
   * @return The function.
   */
  Term synthFunHelper(const std::string& symbol,
                      const std::vector& boundVars,
                      const Sort& sort,
                      bool isInv = false,
                      Grammar* grammar = nullptr) const;

  /** Helper for getting timeout cores */
  std::pair> getTimeoutCoreHelper(
      const std::vector& assumptions) const;
  /**
   * Get value helper, which accounts for subtyping.
   * @param term The term to get the value from.
   * @return The value term.
   */
  Term getValueHelper(const Term& term) const;

  /**
   * Check that the given term is a valid closed term, which can be used as an
   * argument to, e.g., assert, get-value, block-model-values, etc.
   *
   * @param t The term to check.
   */
  void ensureWellFormedTerm(const Term& t) const;
  /** Vector version of above. */
  void ensureWellFormedTerms(const std::vector& ts) const;

  /** Keep a copy of the original option settings (for resets). */
  std::unique_ptr d_originalOptions;
  /** The SMT engine of this solver. */
  std::unique_ptr d_slv;
  /** The random number generator of this solver. */
  std::unique_ptr d_rng;

  /** The associated term manager. */
  TermManager& d_tm;
};

}  // namespace cvc5

#endif




© 2015 - 2024 Weber Informatics LLC | Privacy Policy