LCOV - code coverage report
Current view: top level - buildbot/coverage/build/src/prop - cadical.cpp (source / functions) Hit Total Coverage
Test: coverage.info Lines: 493 571 86.3 %
Date: 2025-03-28 11:57:18 Functions: 62 73 84.9 %
Branches: 242 452 53.5 %

           Branch data     Line data    Source code
       1                 :            : /******************************************************************************
       2                 :            :  * Top contributors (to current version):
       3                 :            :  *   Mathias Preiner, Aina Niemetz, Andrew Reynolds
       4                 :            :  *
       5                 :            :  * This file is part of the cvc5 project.
       6                 :            :  *
       7                 :            :  * Copyright (c) 2009-2025 by the authors listed in the file AUTHORS
       8                 :            :  * in the top-level source directory and their institutional affiliations.
       9                 :            :  * All rights reserved.  See the file COPYING in the top-level source
      10                 :            :  * directory for licensing information.
      11                 :            :  * ****************************************************************************
      12                 :            :  *
      13                 :            :  * Wrapper for CaDiCaL SAT Solver.
      14                 :            :  *
      15                 :            :  * Implementation of the CaDiCaL SAT solver for cvc5 (bit-vectors).
      16                 :            :  */
      17                 :            : 
      18                 :            : #include "prop/cadical.h"
      19                 :            : 
      20                 :            : #include <deque>
      21                 :            : 
      22                 :            : #include "base/check.h"
      23                 :            : #include "options/base_options.h"
      24                 :            : #include "options/main_options.h"
      25                 :            : #include "options/proof_options.h"
      26                 :            : #include "prop/theory_proxy.h"
      27                 :            : #include "util/resource_manager.h"
      28                 :            : #include "util/statistics_registry.h"
      29                 :            : #include "util/string.h"
      30                 :            : 
      31                 :            : namespace cvc5::internal {
      32                 :            : namespace prop {
      33                 :            : 
      34                 :            : /* -------------------------------------------------------------------------- */
      35                 :            : 
      36                 :            : using CadicalLit = int;
      37                 :            : using CadicalVar = int;
      38                 :            : 
      39                 :            : // helper functions
      40                 :            : namespace {
      41                 :            : 
      42                 :      68903 : SatValue toSatValue(int result)
      43                 :            : {
      44         [ +  + ]:      68903 :   if (result == 10) return SAT_VALUE_TRUE;
      45         [ +  - ]:      20093 :   if (result == 20) return SAT_VALUE_FALSE;
      46                 :          0 :   Assert(result == 0);
      47                 :          0 :   return SAT_VALUE_UNKNOWN;
      48                 :            : }
      49                 :            : 
      50                 :            : // Note: CaDiCaL returns lit/-lit for true/false. Older versions returned 1/-1.
      51                 :     553144 : SatValue toSatValueLit(int value)
      52                 :            : {
      53         [ +  + ]:     553144 :   if (value > 0) return SAT_VALUE_TRUE;
      54 [ -  + ][ -  + ]:     341759 :   Assert(value < 0);
                 [ -  - ]
      55                 :     341759 :   return SAT_VALUE_FALSE;
      56                 :            : }
      57                 :            : 
      58                 :   30964500 : CadicalLit toCadicalLit(const SatLiteral lit)
      59                 :            : {
      60         [ +  + ]:   30964500 :   return lit.isNegated() ? -lit.getSatVariable() : lit.getSatVariable();
      61                 :            : }
      62                 :            : 
      63                 :        469 : SatLiteral toSatLiteral(CadicalLit lit)
      64                 :            : {
      65                 :        469 :   return SatLiteral(std::abs(lit), lit < 0);
      66                 :            : }
      67                 :            : 
      68                 :      80391 : CadicalVar toCadicalVar(SatVariable var) { return var; }
      69                 :            : 
      70                 :            : }  // namespace helper functions
      71                 :            : 
      72                 :            : class CadicalPropagator : public CaDiCaL::ExternalPropagator,
      73                 :            :                           public CaDiCaL::FixedAssignmentListener
      74                 :            : {
      75                 :            :  public:
      76                 :         15 :   CadicalPropagator(prop::TheoryProxy* proxy,
      77                 :            :                     context::Context* context,
      78                 :            :                     CaDiCaL::Solver& solver,
      79                 :            :                     StatisticsRegistry& stats)
      80                 :         15 :       : d_proxy(proxy), d_context(*context), d_solver(solver), d_stats(stats)
      81                 :            :   {
      82                 :         15 :     d_var_info.emplace_back();  // 0: Not used
      83                 :         15 :   }
      84                 :            : 
      85                 :            :   /**
      86                 :            :    * Notification from the SAT solver on assignment of a new literal.
      87                 :            :    *
      88                 :            :    * Saves assignment for notified literal, enqueues corresponding theory
      89                 :            :    * literal in theory proxy.
      90                 :            :    *
      91                 :            :    * @param lit      The CaDiCaL literal that was assigned.
      92                 :            :    */
      93                 :        239 :   void notify_assignment(const std::vector<int>& lits) override
      94                 :            :   {
      95 [ +  - ][ -  + ]:        239 :     if (Trace("cadical::propagator").isConnected())
      96                 :            :     {
      97         [ -  - ]:          0 :       Trace("cadical::propagator") << "notif::assignments: { ";
      98         [ -  - ]:          0 :       for (auto lit : lits)
      99                 :            :       {
     100         [ -  - ]:          0 :         Trace("cadical::propagator") << lit << " ";
     101                 :            :       }
     102         [ -  - ]:          0 :       Trace("cadical::propagator") << "}" << std::endl;
     103                 :            :     }
     104                 :        239 :     ++d_stats.notifyAssignment;
     105                 :            :     
     106         [ +  + ]:        239 :     if (d_found_solution)
     107                 :            :     {
     108                 :          2 :       return;
     109                 :            :     }
     110                 :            : 
     111         [ +  + ]:        555 :     for (const auto& lit : lits)
     112                 :            :     {
     113                 :        318 :       SatLiteral slit = toSatLiteral(lit);
     114                 :        318 :       SatVariable var = slit.getSatVariable();
     115 [ -  + ][ -  + ]:        318 :       Assert(var < d_var_info.size());
                 [ -  - ]
     116                 :            : 
     117                 :        318 :       auto& info = d_var_info[var];
     118                 :            : 
     119                 :            :       // Only consider active variables
     120         [ -  + ]:        318 :       if (!info.is_active)
     121                 :            :       {
     122                 :          0 :         continue;
     123                 :            :       }
     124                 :            : 
     125                 :        318 :       bool is_decision = d_solver.is_decision(lit);
     126                 :            : 
     127         [ +  - ]:        636 :       Trace("cadical::propagator")
     128         [ -  - ]:          0 :           << "notif::assignment: [" << (is_decision ? "d" : "p") << "] " << slit
     129                 :          0 :           << " (level: " << d_decisions.size()
     130                 :          0 :           << ", level_intro: " << info.level_intro
     131                 :        318 :           << ", level_user: " << current_user_level() << ")" << std::endl;
     132                 :            : 
     133                 :            :       // Save decision variables
     134         [ +  + ]:        318 :       if (is_decision)
     135                 :            :       {
     136                 :         38 :         d_decisions.back() = slit;
     137                 :            :       }
     138                 :            : 
     139 [ +  + ][ -  + ]:        318 :       Assert(info.assignment == 0 || info.assignment == lit);
         [ -  + ][ -  - ]
     140                 :            : 
     141                 :            :       // Only notify theory proxy if variable was assigned a new value, not if
     142                 :            :       // it got fixed after assignment already happend.
     143         [ +  + ]:        318 :       if (info.assignment == 0)
     144                 :            :       {
     145                 :        213 :         info.assignment = lit;
     146                 :        213 :         d_assignments.push_back(slit);
     147         [ +  + ]:        213 :         if (info.is_theory_atom)
     148                 :            :         {
     149         [ +  - ]:        130 :           Trace("cadical::propagator") << "enqueue: " << slit << std::endl;
     150         [ +  - ]:        260 :           Trace("cadical::propagator")
     151 [ -  + ][ -  - ]:        130 :               << "node:    " << d_proxy->getNode(slit) << std::endl;
     152                 :        130 :           d_proxy->enqueueTheoryLiteral(slit);
     153                 :            :         }
     154                 :            :       }
     155                 :            :     }
     156                 :            :   }
     157                 :            : 
     158                 :            :   /**
     159                 :            :    * Notification from the SAT solver on fixed assignment of a literal.
     160                 :            :    *
     161                 :            :    * Notifications on fixed assignment are sent during the search, while
     162                 :            :    * regular assignment notifications are sent batch-wise, before the first
     163                 :            :    * callback is issued. This thus calls notify_assignment() to trigger what
     164                 :            :    * needs to be done on assignment. Already notified fixed assignments are
     165                 :            :    * then skipped in notify_assignment().
     166                 :            :    *
     167                 :            :    * @param lit The CaDiCaL literal that was fixed.
     168                 :            :    */
     169                 :        139 :   void notify_fixed_assignment(int lit) override
     170                 :            :   {
     171                 :        139 :     SatLiteral slit = toSatLiteral(lit);
     172                 :        139 :     SatVariable var = slit.getSatVariable();
     173                 :            : 
     174                 :            :     // We don't care about non-observed variables
     175         [ -  + ]:        139 :     if (var >= d_var_info.size())
     176                 :            :     {
     177                 :         32 :       return;
     178                 :            :     }
     179                 :            : 
     180                 :        139 :     auto& info = d_var_info[var];
     181                 :            :     // Only consider active variables
     182         [ +  + ]:        139 :     if (!info.is_active)
     183                 :            :     {
     184                 :         32 :       return;
     185                 :            :     }
     186                 :        107 :     ++d_stats.notifyFixedAssignment;
     187                 :            : 
     188         [ +  - ]:        214 :     Trace("cadical::propagator")
     189                 :        107 :         << "notif::fixed assignment: " << slit << std::endl;
     190                 :            : 
     191                 :            :     // Mark as fixed.
     192 [ -  + ][ -  + ]:        107 :     Assert(!info.is_fixed);
                 [ -  - ]
     193                 :        107 :     info.is_fixed = true;
     194                 :        107 :     info.level_fixed = current_user_level();
     195                 :            :     // Trigger actual assignment.
     196                 :        107 :     notify_assignment({lit});
     197                 :            :   }
     198                 :            : 
     199                 :            :   /**
     200                 :            :    * Notification from the SAT solver when it makes a decision.
     201                 :            :    * Pushes new SAT context level.
     202                 :            :    */
     203                 :         40 :   void notify_new_decision_level() override
     204                 :            :   {
     205                 :         40 :     d_context.push();
     206                 :         40 :     d_assignment_control.push_back(d_assignments.size());
     207                 :         40 :     d_decisions.emplace_back();
     208         [ +  - ]:         80 :     Trace("cadical::propagator")
     209                 :         40 :         << "notif::decision: new level " << d_decisions.size() << std::endl;
     210                 :         40 :     ++d_stats.notifyNewDecision;
     211                 :         40 :   }
     212                 :            : 
     213                 :            :   /**
     214                 :            :    * Notification from the SAT solver on backtrack to the given level.
     215                 :            :    *
     216                 :            :    * This will automatically backtrack decisions and assignments to the
     217                 :            :    * specified level. Fixed assignments that get backtracked will be
     218                 :            :    * re-assigned at `level` and the corresponding theory literals are
     219                 :            :    * re-enqueued in the theory proxy.
     220                 :            :    *
     221                 :            :    * @param level The level the SAT solver backtracked to.
     222                 :            :    */
     223                 :         62 :   void notify_backtrack(size_t level) override
     224                 :            :   {
     225         [ +  - ]:         62 :     Trace("cadical::propagator") << "notif::backtrack: " << level << std::endl;
     226                 :            : 
     227                 :            :     // CaDiCaL may notify us about backtracks of decisions that we were not
     228                 :            :     // notified about. We can safely ignore them.
     229         [ +  + ]:         62 :     if (d_decisions.size() <= level)
     230                 :            :     {
     231 [ -  + ][ -  + ]:         30 :       Assert(d_decisions.size() == 0);
                 [ -  - ]
     232                 :         30 :       return;
     233                 :            :     }
     234                 :         32 :     d_found_solution = false;
     235                 :            : 
     236                 :            :     // Backtrack decisions
     237 [ -  + ][ -  + ]:         32 :     Assert(d_decisions.size() > level);
                 [ -  - ]
     238 [ -  + ][ -  + ]:         32 :     Assert(d_context.getLevel() > level);
                 [ -  - ]
     239         [ +  + ]:         72 :     for (size_t cur_level = d_decisions.size(); cur_level > level; --cur_level)
     240                 :            :     {
     241                 :         40 :       d_context.pop();
     242                 :         40 :       d_decisions.pop_back();
     243                 :            :     }
     244                 :            : 
     245                 :            :     // Backtrack assignments, resend fixed theory literals that got backtracked
     246 [ -  + ][ -  + ]:         32 :     Assert(!d_assignment_control.empty());
                 [ -  - ]
     247                 :         32 :     size_t pop_to = d_assignment_control[level];
     248                 :         32 :     d_assignment_control.resize(level);
     249                 :            : 
     250         [ +  + ]:        138 :     while (pop_to < d_assignments.size())
     251                 :            :     {
     252                 :        106 :       SatLiteral lit = d_assignments.back();
     253                 :        106 :       d_assignments.pop_back();
     254                 :        106 :       SatVariable var = lit.getSatVariable();
     255                 :        106 :       auto& info = d_var_info[var];
     256         [ +  - ]:        106 :       Trace("cadical::propagator") << "unassign: " << var << std::endl;
     257                 :        106 :       info.assignment = 0;
     258                 :            :     }
     259                 :            : 
     260                 :            :     // Notify theory proxy about backtrack
     261                 :         32 :     d_proxy->notifyBacktrack();
     262                 :            :     // Clear the propgations since they are not valid anymore.
     263                 :         32 :     d_propagations.clear();
     264                 :         32 :     ++d_stats.notifyBacktrack;
     265                 :            : 
     266         [ +  - ]:         32 :     Trace("cadical::propagator") << "notif::backtrack end" << std::endl;
     267                 :            :   }
     268                 :            : 
     269                 :            :   /**
     270                 :            :    * Callback of the SAT solver on finding a full sat assignment.
     271                 :            :    *
     272                 :            :    * Checks whether current model is consistent with the theories, performs a
     273                 :            :    * full effort check and theory propgations.
     274                 :            :    *
     275                 :            :    * @param model The full assignment.
     276                 :            :    * @return true If the current model is not in conflict with the theories.
     277                 :            :    */
     278                 :         55 :   bool cb_check_found_model(const std::vector<int>& model) override
     279                 :            :   {
     280         [ +  - ]:         55 :     Trace("cadical::propagator") << "cb::check_found_model" << std::endl;
     281                 :         55 :     bool recheck = false;
     282                 :            : 
     283         [ +  + ]:         55 :     if (d_found_solution)
     284                 :            :     {
     285                 :          2 :       return true;
     286                 :            :     }
     287                 :            : 
     288                 :         53 :     ++d_stats.cbCheckFoundModel;
     289                 :            :     // CaDiCaL may backtrack while importing clauses, which can result in some
     290                 :            :     // clauses not being processed. Make sure to add all clauses before
     291                 :            :     // checking the model.
     292         [ +  + ]:         53 :     if (!d_new_clauses.empty())
     293                 :            :     {
     294         [ +  - ]:          8 :       Trace("cadical::propagator") << "cb::check_found_model end: new "
     295                 :          0 :                                       "variables added via theory decision"
     296                 :          4 :                                    << std::endl;
     297                 :            :       // CaDiCaL expects us to be able to provide a reason for rejecting the
     298                 :            :       // model (it asserts that after this call, cb_has_external_clause()
     299                 :            :       // returns true). However, in this particular case, we want to force
     300                 :            :       // CaDiCaL to give us model values for the new variables that were
     301                 :            :       // introduced (to kick off the assignment notification machinery), we
     302                 :            :       // don't have a reason clause for rejecting the model. CaDiCaL's
     303                 :            :       // expectation will be weakened in the future to allow for this, but for
     304                 :            :       // now we simply add a tautology as reason to pacify CaDiCaL.
     305                 :          4 :       d_new_clauses.push_back(1);
     306                 :          4 :       d_new_clauses.push_back(-1);
     307                 :          4 :       d_new_clauses.push_back(0);
     308                 :          4 :       return false;
     309                 :            :     }
     310                 :            : 
     311                 :            :     // Check full model.
     312                 :            :     //
     313                 :            :     // First, we have to ensure that if the SAT solver determines sat without
     314                 :            :     // making any decisions, theory decisions are still requested until fixed
     315                 :            :     // point at least once since some modules, e.g., finite model finding, rely
     316                 :            :     // on this. Theory decisions may add new variables (while decisions
     317                 :            :     // requested by the decision engine will not). If new variables are added,
     318                 :            :     // we interrupt the check to force the SAT solver to extend the model with
     319                 :            :     // the new variables.
     320                 :         49 :     size_t size = d_var_info.size();
     321                 :            :     bool requirePhase, stopSearch;
     322                 :         49 :     d_proxy->getNextDecisionRequest(requirePhase, stopSearch);
     323         [ -  + ]:         49 :     if (d_var_info.size() != size)
     324                 :            :     {
     325         [ -  - ]:          0 :       Trace("cadical::propagator") << "cb::check_found_model end: new "
     326                 :          0 :                                       "variables added via theory decision"
     327                 :          0 :                                    << std::endl;
     328                 :          0 :       return false;
     329                 :            :     }
     330                 :            :     // Theory engine may trigger a recheck, unless new variables were added
     331                 :            :     // during check. If so, we break out of the check and have the SAT solver
     332                 :            :     // extend the model with the new variables.
     333                 :          2 :     do
     334                 :            :     {
     335         [ +  - ]:        102 :       Trace("cadical::propagator")
     336                 :         51 :           << "full check (recheck: " << recheck << ")" << std::endl;
     337                 :         51 :       d_proxy->theoryCheck(theory::Theory::Effort::EFFORT_FULL);
     338                 :         51 :       theory_propagate();
     339         [ +  + ]:         59 :       for (const SatLiteral& p : d_propagations)
     340                 :            :       {
     341         [ +  - ]:         16 :         Trace("cadical::propagator")
     342                 :          8 :             << "add propagation reason: " << p << std::endl;
     343                 :         16 :         SatClause clause;
     344                 :          8 :         d_proxy->explainPropagation(p, clause);
     345                 :          8 :         add_clause(clause);
     346                 :            :       }
     347                 :         51 :       d_propagations.clear();
     348                 :            : 
     349         [ +  + ]:         51 :       if (!d_new_clauses.empty())
     350                 :            :       {
     351                 :            :         // Will again call cb_check_found_model() after clauses were added.
     352                 :         33 :         recheck = false;
     353                 :            :       }
     354                 :            :       else
     355                 :            :       {
     356                 :         18 :         recheck = d_proxy->theoryNeedCheck();
     357                 :            :       }
     358 [ +  + ][ +  + ]:         51 :     } while (d_var_info.size() == size && recheck);
                 [ +  + ]
     359                 :            : 
     360         [ +  + ]:         49 :     if (d_var_info.size() != size)
     361                 :            :     {
     362         [ +  - ]:         36 :       Trace("cadical::propagator") << "cb::check_found_model end: new "
     363                 :          0 :                                       "variables added via theory check"
     364                 :         18 :                                    << std::endl;
     365                 :            :       // Same as above, until CaDiCaL's assertion that we have to have
     366                 :            :       // a reason clause for rejecting the model is weakened, we need to
     367                 :            :       // pacify it with a tautology.
     368                 :         18 :       d_new_clauses.push_back(1);
     369                 :         18 :       d_new_clauses.push_back(-1);
     370                 :         18 :       d_new_clauses.push_back(0);
     371                 :         18 :       return false;
     372                 :            :     }
     373                 :         31 :     bool res = done();
     374         [ +  - ]:         62 :     Trace("cadical::propagator")
     375                 :         31 :         << "cb::check_found_model end: done: " << res << std::endl;
     376                 :         31 :     return res;
     377                 :            :   }
     378                 :            : 
     379                 :            :   /**
     380                 :            :    * Callback of the SAT solver before making a new decision.
     381                 :            :    *
     382                 :            :    * Processes decision requests from the theory proxy.
     383                 :            :    *
     384                 :            :    * @note This may call cb_check_found_model() in case the decision engine
     385                 :            :    *       determines that we have a partial model, i.e., stopSearch is true.
     386                 :            :    *
     387                 :            :    * @return The next decision.
     388                 :            :    */
     389                 :         26 :   int cb_decide() override
     390                 :            :   {
     391         [ +  - ]:         26 :     Trace("cadical::propagator") << "cb::decide" << std::endl;
     392         [ -  + ]:         26 :     if (d_found_solution)
     393                 :            :     {
     394                 :          0 :       return 0;
     395                 :            :     }
     396                 :         26 :     ++d_stats.cbDecide;
     397                 :         26 :     bool stopSearch = false;
     398                 :         26 :     bool requirePhase = false;
     399                 :         26 :     SatLiteral lit = d_proxy->getNextDecisionRequest(requirePhase, stopSearch);
     400                 :            :     // We found a partial model, let's check it.
     401         [ +  + ]:         26 :     if (stopSearch)
     402                 :            :     {
     403                 :         12 :       d_found_solution = cb_check_found_model({});
     404         [ +  + ]:         12 :       if (d_found_solution)
     405                 :            :       {
     406         [ +  - ]:          2 :         Trace("cadical::propagator") << "Found solution" << std::endl;
     407                 :          2 :         d_found_solution = d_proxy->isDecisionEngineDone();
     408         [ -  + ]:          2 :         if (!d_found_solution)
     409                 :            :         {
     410         [ -  - ]:          0 :           Trace("cadical::propagator")
     411                 :          0 :               << "Decision engine not done" << std::endl;
     412                 :          0 :           lit = d_proxy->getNextDecisionRequest(requirePhase, stopSearch);
     413                 :            :         }
     414                 :            :       }
     415                 :            :       else
     416                 :            :       {
     417         [ +  - ]:         10 :         Trace("cadical::propagator") << "No solution found yet" << std::endl;
     418                 :            :       }
     419                 :            :     }
     420 [ +  + ][ +  - ]:         26 :     if (!stopSearch && lit != undefSatLiteral)
                 [ +  + ]
     421                 :            :     {
     422         [ +  + ]:         14 :       if (!requirePhase)
     423                 :            :       {
     424                 :          8 :         int8_t phase = d_var_info[lit.getSatVariable()].phase;
     425         [ -  + ]:          8 :         if (phase != 0)
     426                 :            :         {
     427         [ -  - ]:          0 :           if ((phase == -1 && !lit.isNegated())
     428 [ -  - ][ -  - ]:          0 :               || (phase == 1 && lit.isNegated()))
         [ -  - ][ -  - ]
     429                 :            :           {
     430                 :          0 :             lit = ~lit;
     431                 :            :           }
     432                 :            :         }
     433                 :            :       }
     434         [ +  - ]:         14 :       Trace("cadical::propagator") << "cb::decide: " << lit << std::endl;
     435                 :         14 :       return toCadicalLit(lit);
     436                 :            :     }
     437         [ +  - ]:         12 :     Trace("cadical::propagator") << "cb::decide: 0" << std::endl;
     438                 :         12 :     return 0;
     439                 :            :   }
     440                 :            : 
     441                 :            :   /**
     442                 :            :    * Callback of the SAT solver after BCP.
     443                 :            :    *
     444                 :            :    * Performs standard theory check and theory propagations.
     445                 :            :    *
     446                 :            :    * If we don't have any theory propagations queued in d_propagations, we
     447                 :            :    * perform an EFFORT_STANDARD check in combination with theory_propagate() to
     448                 :            :    * populate d_propagations.
     449                 :            :    *
     450                 :            :    * @return The next theory propagation.
     451                 :            :    */
     452                 :        158 :   int cb_propagate() override
     453                 :            :   {
     454         [ +  + ]:        158 :     if (d_found_solution)
     455                 :            :     {
     456                 :          2 :       return 0;
     457                 :            :     }
     458                 :        156 :     ++d_stats.cbPropagate;
     459         [ +  - ]:        156 :     Trace("cadical::propagator") << "cb::propagate" << std::endl;
     460         [ +  + ]:        156 :     if (d_propagations.empty())
     461                 :            :     {
     462                 :            :       // Only propagate if all activation literals are processed. Activation
     463                 :            :       // literals are always assumed first. If we don't do this, explanations
     464                 :            :       // for theory propgations may force activation literals to different
     465                 :            :       // values before they can get decided on.
     466         [ +  + ]:        135 :       if (d_decisions.size() < current_user_level())
     467                 :            :       {
     468                 :         26 :         return 0;
     469                 :            :       }
     470                 :        109 :       d_proxy->theoryCheck(theory::Theory::Effort::EFFORT_STANDARD);
     471                 :        109 :       theory_propagate();
     472                 :            :     }
     473                 :        130 :     return next_propagation();
     474                 :            :   }
     475                 :            : 
     476                 :            :   /**
     477                 :            :    * Callback of the SAT solver asking for the explanation of a theory literal.
     478                 :            :    * @note This is called on `propagated_lit` until the reason clause is
     479                 :            :    *       fully processed.
     480                 :            :    * @param propagated_lit The theory literal.
     481                 :            :    * @return The next literal of the reason clause, 0 to terminate the clause.
     482                 :            :    */
     483                 :         41 :   int cb_add_reason_clause_lit(int propagated_lit) override
     484                 :            :   {
     485                 :         41 :     ++d_stats.cbAddReasonClauseLit;
     486                 :            :     // Get reason for propagated_lit.
     487         [ +  + ]:         41 :     if (!d_processing_reason)
     488                 :            :     {
     489 [ -  + ][ -  + ]:         11 :       Assert(d_reason.empty());
                 [ -  - ]
     490                 :         11 :       SatLiteral slit = toSatLiteral(propagated_lit);
     491                 :         11 :       SatClause clause;
     492                 :         11 :       d_proxy->explainPropagation(slit, clause);
     493                 :            :       // Add activation literal to reason
     494                 :         11 :       SatLiteral alit = current_activation_lit();
     495         [ +  + ]:         11 :       if (alit != undefSatLiteral)
     496                 :            :       {
     497                 :          4 :         d_reason.push_back(alit);
     498                 :            :       }
     499                 :         11 :       d_reason.insert(d_reason.end(), clause.begin(), clause.end());
     500                 :         11 :       d_processing_reason = true;
     501         [ +  - ]:         22 :       Trace("cadical::propagator")
     502                 :         11 :           << "cb::reason: " << slit << ", size: " << d_reason.size()
     503                 :         11 :           << std::endl;
     504                 :            :     }
     505                 :            : 
     506                 :            :     // We are done processing the reason for propagated_lit.
     507         [ +  + ]:         41 :     if (d_reason.empty())
     508                 :            :     {
     509                 :         11 :       d_processing_reason = false;
     510                 :         11 :       return 0;
     511                 :            :     }
     512                 :            : 
     513                 :            :     // Return next literal of the reason for propagated_lit.
     514         [ +  - ]:         60 :     Trace("cadical::propagator")
     515                 :         30 :         << "cb::reason: " << toSatLiteral(propagated_lit) << " "
     516                 :         30 :         << d_reason.front() << std::endl;
     517                 :         30 :     int lit = toCadicalLit(d_reason.front());
     518                 :         30 :     d_reason.pop_front();
     519                 :         30 :     return lit;
     520                 :            :   }
     521                 :            : 
     522                 :            :   /**
     523                 :            :    * Callback of the SAT solver to determine if we have a new clause to add.
     524                 :            :    * @param forgettable True if clause is not irredundant.
     525                 :            :    * @return True to indicate that we have clauses to add.
     526                 :            :    */
     527                 :        178 :   bool cb_has_external_clause(bool& is_forgettable) override
     528                 :            :   {
     529                 :        178 :     ++d_stats.cbHasExternalClause;
     530                 :        178 :     is_forgettable = false;
     531                 :        178 :     return !d_new_clauses.empty();
     532                 :            :   }
     533                 :            : 
     534                 :            :   /**
     535                 :            :    * Callback of the SAT solver to add a new clause.
     536                 :            :    * @note This is called consecutively until the full clause is processed.
     537                 :            :    * @note Clauses are terminated with 0 in d_new_clauses.
     538                 :            :    * @return The next literal of the clause, 0 to terminate the clause.
     539                 :            :    */
     540                 :        251 :   int cb_add_external_clause_lit() override
     541                 :            :   {
     542                 :        251 :     ++d_stats.cbAddExternalClauseLit;
     543 [ -  + ][ -  + ]:        251 :     Assert(!d_new_clauses.empty());
                 [ -  - ]
     544                 :        251 :     CadicalLit lit = d_new_clauses.front();
     545                 :        251 :     d_new_clauses.pop_front();
     546         [ +  - ]:        502 :     Trace("cadical::propagator")
     547                 :        251 :         << "external_clause: " << toSatLiteral(lit) << std::endl;
     548                 :        251 :     return lit;
     549                 :            :   }
     550                 :            : 
     551                 :            :   /**
     552                 :            :    * Get the current trail of decisions.
     553                 :            :    * @return The trail of decisions.
     554                 :            :    */
     555                 :          0 :   const std::vector<SatLiteral>& get_decisions() const { return d_decisions; }
     556                 :            : 
     557                 :            :   /**
     558                 :            :    * Get the current assignment of lit.
     559                 :            :    *
     560                 :            :    * Note: This does not query d_solver->val() since this can only be queried
     561                 :            :    * if the SAT solver is in a SAT state, which is not the case during solving.
     562                 :            :    *
     563                 :            :    * @param lit SatLiteral to be queried.
     564                 :            :    * @return Current value of given literal on the trail.
     565                 :            :    */
     566                 :        401 :   SatValue value(SatLiteral lit) const
     567                 :            :   {
     568                 :        401 :     SatVariable var = lit.getSatVariable();
     569                 :        401 :     SatValue val = SAT_VALUE_UNKNOWN;
     570                 :        401 :     int32_t assign = d_var_info[var].assignment;
     571         [ +  + ]:        401 :     if (assign != 0)
     572                 :            :     {
     573         [ +  + ]:        258 :       val = toSatValueLit(lit.isNegated() ? -assign : assign);
     574                 :            :     }
     575         [ +  - ]:        802 :     Trace("cadical::propagator")
     576                 :        401 :         << "value: " << lit << ": " << val << std::endl;
     577                 :        401 :     return val;
     578                 :            :   }
     579                 :            : 
     580                 :            :   /**
     581                 :            :    * Adds a new clause to the propagator.
     582                 :            :    *
     583                 :            :    * The clause will not immediately added to the SAT solver, but instead
     584                 :            :    * will be added through the `cb_add_external_clause_lit` callback.
     585                 :            :    *
     586                 :            :    * Note: Filters out clauses satisfied by fixed literals.
     587                 :            :    *
     588                 :            :    * @param clause The clause to add.
     589                 :            :    */
     590                 :        215 :   void add_clause(const SatClause& clause)
     591                 :            :   {
     592                 :        215 :     std::vector<CadicalLit> lits;
     593         [ +  + ]:        424 :     for (const SatLiteral& lit : clause)
     594                 :            :     {
     595                 :        302 :       SatVariable var = lit.getSatVariable();
     596 [ -  + ][ -  + ]:        302 :       Assert(var < d_var_info.size());
                 [ -  - ]
     597                 :        302 :       const auto& info = d_var_info[var];
     598 [ -  + ][ -  + ]:        302 :       Assert(info.is_active);
                 [ -  - ]
     599         [ +  + ]:        302 :       if (info.is_fixed)
     600                 :            :       {
     601         [ +  + ]:        138 :         int32_t val = lit.isNegated() ? -info.assignment : info.assignment;
     602 [ -  + ][ -  + ]:        138 :         Assert(val != 0);
                 [ -  - ]
     603         [ +  + ]:        138 :         if (val > 0)
     604                 :            :         {
     605                 :            :           // Clause satisfied by fixed literal, no clause added
     606                 :         93 :           return;
     607                 :            :         }
     608                 :            :       }
     609                 :        209 :       lits.push_back(toCadicalLit(lit));
     610                 :            :     }
     611         [ +  - ]:        122 :     if (!lits.empty())
     612                 :            :     {
     613                 :            :       // Add activation literal to clause if we are in user level > 0
     614                 :        122 :       SatLiteral alit = current_activation_lit();
     615         [ +  + ]:        122 :       if (alit != undefSatLiteral)
     616                 :            :       {
     617                 :         38 :         lits.insert(lits.begin(), toCadicalLit(alit));
     618                 :            :       }
     619                 :            :       // Do not immediately add clauses added during search. We have to buffer
     620                 :            :       // them and add them during the cb_add_reason_clause_lit callback.
     621         [ +  + ]:        122 :       if (d_in_search)
     622                 :            :       {
     623                 :         55 :         d_new_clauses.insert(d_new_clauses.end(), lits.begin(), lits.end());
     624                 :         55 :         d_new_clauses.push_back(0);
     625                 :            :       }
     626                 :            :       else
     627                 :            :       {
     628         [ +  + ]:        168 :         for (const auto& lit : lits)
     629                 :            :         {
     630                 :        101 :           d_solver.add(lit);
     631                 :            :         }
     632                 :         67 :         d_solver.add(0);
     633                 :            :       }
     634                 :            :     }
     635                 :            :   }
     636                 :            : 
     637                 :            :   /**
     638                 :            :    * Add new CaDiCaL variable.
     639                 :            :    * @param var            The variable to add.
     640                 :            :    * @param level          The current user assertion level.
     641                 :            :    * @param is_theory_atom True if variable is a theory atom.
     642                 :            :    * @param in_search      True if SAT solver is currently in search().
     643                 :            :    */
     644                 :        150 :   void add_new_var(const SatVariable& var, bool is_theory_atom)
     645                 :            :   {
     646                 :            :     // Since activation literals are not tracked here, we have to make sure to
     647                 :            :     // properly resize d_var_info.
     648         [ -  + ]:        150 :     if (var > d_var_info.size())
     649                 :            :     {
     650                 :          0 :       d_var_info.resize(var);
     651                 :            :     }
     652 [ -  + ][ -  + ]:        150 :     Assert(d_var_info.size() == var);
                 [ -  - ]
     653                 :            : 
     654                 :            :     // Boolean variables are not theory atoms, but may still occur in
     655                 :            :     // lemmas/conflicts sent to the SAT solver. Hence, we have to observe them
     656                 :            :     // since CaDiCaL expects all literals sent back to be observed.
     657                 :        150 :     d_solver.add_observed_var(toCadicalVar(var));
     658                 :        150 :     d_active_vars.push_back(var);
     659         [ +  - ]:        300 :     Trace("cadical::propagator")
     660                 :        150 :         << "new var: " << var << " (level: " << current_user_level()
     661                 :          0 :         << ", is_theory_atom: " << is_theory_atom
     662                 :        150 :         << ", in_search: " << d_in_search << ")" << std::endl;
     663                 :        150 :     auto& info = d_var_info.emplace_back();
     664                 :        150 :     info.level_intro = current_user_level();
     665                 :        150 :     info.is_theory_atom = is_theory_atom;
     666                 :        150 :   }
     667                 :            : 
     668                 :            :   /**
     669                 :            :    * Checks whether the theory engine is done, no new clauses need to be added
     670                 :            :    * and the current model is consistent.
     671                 :            :    */
     672                 :         47 :   bool done() const
     673                 :            :   {
     674         [ +  + ]:         47 :     if (!d_new_clauses.empty())
     675                 :            :     {
     676         [ +  - ]:         15 :       Trace("cadical::propagator") << "not done: pending clauses" << std::endl;
     677                 :         15 :       return false;
     678                 :            :     }
     679         [ -  + ]:         32 :     if (d_proxy->theoryNeedCheck())
     680                 :            :     {
     681         [ -  - ]:          0 :       Trace("cadical::propagator")
     682                 :          0 :           << "not done: theory need check" << std::endl;
     683                 :          0 :       return false;
     684                 :            :     }
     685         [ +  + ]:         32 :     if (d_found_solution)
     686                 :            :     {
     687         [ +  - ]:          4 :       Trace("cadical::propagator")
     688                 :          2 :           << "done: found partial solution" << std::endl;
     689                 :            :     }
     690                 :            :     else
     691                 :            :     {
     692         [ +  - ]:         60 :       Trace("cadical::propagator")
     693                 :         30 :           << "done: full assignment consistent" << std::endl;
     694                 :            :     }
     695                 :         32 :     return true;
     696                 :            :   }
     697                 :            : 
     698                 :            :   /**
     699                 :            :    * Push user assertion level.
     700                 :            :    */
     701                 :         20 :   void user_push()
     702                 :            :   {
     703         [ +  - ]:         40 :     Trace("cadical::propagator")
     704                 :         20 :         << "user push: " << d_active_vars_control.size();
     705                 :         20 :     d_active_vars_control.push_back(d_active_vars.size());
     706         [ +  - ]:         40 :     Trace("cadical::propagator")
     707                 :         20 :         << " -> " << d_active_vars_control.size() << std::endl;
     708                 :         20 :   }
     709                 :            : 
     710                 :            :   /**
     711                 :            :    * Set the activation literal for the current user assertion level.
     712                 :            :    *
     713                 :            :    * @param alit The activation literal for the current user assertion level.
     714                 :            :    */
     715                 :         20 :   void set_activation_lit(SatVariable& alit)
     716                 :            :   {
     717                 :         20 :     d_activation_literals.push_back(alit);
     718         [ +  - ]:         40 :     Trace("cadical::propagator")
     719                 :         20 :       << "enable activation lit: " << alit << std::endl;
     720                 :         20 :   }
     721                 :            : 
     722                 :            :   /**
     723                 :            :    * Pop user assertion level.
     724                 :            :    */
     725                 :         20 :   void user_pop()
     726                 :            :   {
     727         [ +  - ]:         40 :     Trace("cadical::propagator")
     728                 :         20 :         << "user pop: " << d_active_vars_control.size();
     729                 :         20 :     size_t pop_to = d_active_vars_control.back();
     730                 :         20 :     d_active_vars_control.pop_back();
     731         [ +  - ]:         40 :     Trace("cadical::propagator")
     732                 :         20 :         << " -> " << d_active_vars_control.size() << std::endl;
     733                 :            : 
     734                 :            :     // Disable activation literal for popped user level. The activation literal
     735                 :            :     // is added as unit clause, which will satisfy all clauses added in this
     736                 :            :     // user level and get garbage collected in the SAT solver.
     737                 :         20 :     SatLiteral alit = current_activation_lit();
     738         [ +  - ]:         40 :     Trace("cadical::propagator")
     739                 :         20 :         << "disable activation lit: " << alit << std::endl;
     740                 :         20 :     d_activation_literals.pop_back();
     741                 :            : 
     742                 :         20 :     size_t user_level = current_user_level();
     743                 :            : 
     744                 :            :     // Unregister popped variables so that CaDiCaL does not notify us anymore
     745                 :            :     // about assignments.
     746 [ -  + ][ -  + ]:         20 :     Assert(pop_to <= d_active_vars.size());
                 [ -  - ]
     747                 :         40 :     std::vector<SatVariable> fixed;
     748         [ +  + ]:         61 :     while (d_active_vars.size() > pop_to)
     749                 :            :     {
     750                 :         41 :       SatVariable var = d_active_vars.back();
     751                 :         41 :       const auto& info = d_var_info[var];
     752                 :         41 :       d_active_vars.pop_back();
     753                 :            : 
     754                 :            :       // We keep fixed literals that correspond to theory atoms introduced in
     755                 :            :       // lower user levels, since we have to renotify them before the next
     756                 :            :       // solve call.
     757 [ +  + ][ -  + ]:         41 :       if (info.is_fixed && info.is_theory_atom
     758         [ -  - ]:          0 :           && info.level_intro <= user_level)
     759                 :            :       {
     760                 :          0 :         fixed.push_back(var);
     761                 :            :       }
     762                 :            :       else
     763                 :            :       {
     764         [ +  - ]:         41 :         Trace("cadical::propagator") << "set inactive: " << var << std::endl;
     765                 :         41 :         d_var_info[var].is_active = false;
     766                 :         41 :         d_solver.remove_observed_var(toCadicalVar(var));
     767 [ -  + ][ -  + ]:         41 :         Assert(info.level_intro > user_level);
                 [ -  - ]
     768                 :            :         // Fix value of inactive variables in order to avoid CaDiCaL from
     769                 :            :         // deciding on them again. This make a huge difference in performance
     770                 :            :         // for incremental problems with many check-sat calls.
     771                 :         41 :         d_solver.add(toCadicalLit(var));
     772                 :         41 :         d_solver.add(0);
     773                 :            :       }
     774                 :            :     }
     775                 :            :     // Re-add fixed active vars in the order they were added to d_active_vars.
     776                 :         20 :     d_active_vars.insert(d_active_vars.end(), fixed.rbegin(), fixed.rend());
     777                 :            : 
     778                 :            :     // We are at decicion level 0 at this point.
     779 [ -  + ][ -  + ]:         20 :     Assert(d_decisions.empty());
                 [ -  - ]
     780 [ -  + ][ -  + ]:         20 :     Assert(d_assignment_control.empty());
                 [ -  - ]
     781                 :            :     // At this point, only fixed literals will be on d_assignments, now we have
     782                 :            :     // to determine which of these are still relevant in the current user
     783                 :            :     // level. If the variable is still active here, it means that it is still
     784                 :            :     // relevant for the current user level. If its assignment was fixed in a
     785                 :            :     // higher user level, we have to renotify the fixed literal in the current
     786                 :            :     // level (or in the user level of the next solve call). This happens by
     787                 :            :     // pushing the literal onto the d_renotify_fixed vector.
     788                 :         20 :     auto it = d_assignments.begin();
     789         [ +  + ]:        112 :     while (it != d_assignments.end())
     790                 :            :     {
     791                 :         92 :       SatLiteral lit = *it;
     792                 :         92 :       auto& info = d_var_info[lit.getSatVariable()];
     793 [ -  + ][ -  + ]:         92 :       Assert(info.is_fixed);
                 [ -  - ]
     794                 :            : 
     795                 :            :       // Remove inactive variables from the assignment vector.
     796         [ +  + ]:         92 :       if (!info.is_active)
     797                 :            :       {
     798                 :          9 :         it = d_assignments.erase(it);
     799                 :          9 :         continue;
     800                 :            :       }
     801                 :            : 
     802                 :            :       // Renotify fixed literals if it was fixed in a higher user level.
     803 [ +  + ][ -  + ]:         83 :       if (info.is_theory_atom && info.level_fixed > user_level)
     804                 :            :       {
     805         [ -  - ]:          0 :         Trace("cadical::propagator")
     806                 :          0 :             << "queue renotify: " << lit
     807                 :          0 :             << " (level_intro: " << info.level_intro
     808                 :          0 :             << ", level_fixed: " << info.level_fixed << ")" << std::endl;
     809                 :          0 :         d_renotify_fixed.push_back(lit);
     810                 :            :       }
     811                 :         83 :       ++it;
     812                 :            :     }
     813                 :         20 :   }
     814                 :            : 
     815                 :          0 :   bool is_fixed(SatVariable var) const { return d_var_info[var].is_fixed; }
     816                 :            : 
     817                 :            :   /**
     818                 :            :    * Configure and record preferred phase of variable.
     819                 :            :    * @param lit The literal.
     820                 :            :    */
     821                 :          2 :   void phase(SatLiteral lit)
     822                 :            :   {
     823                 :          2 :     d_solver.phase(toCadicalLit(lit));
     824         [ -  + ]:          2 :     d_var_info[lit.getSatVariable()].phase = lit.isNegated() ? -1 : 1;
     825                 :          2 :   }
     826                 :            : 
     827                 :            :   /**
     828                 :            :    * Return the activation literal for the current user level.
     829                 :            :    *
     830                 :            :    * Note: Returns undefSatLiteral at user level 0.
     831                 :            :    */
     832                 :        153 :   const SatLiteral& current_activation_lit()
     833                 :            :   {
     834         [ +  + ]:        153 :     if (d_activation_literals.empty())
     835                 :            :     {
     836                 :         91 :       return undefSatLiteral;
     837                 :            :     }
     838                 :         62 :     return d_activation_literals.back();
     839                 :            :   }
     840                 :            : 
     841                 :            :   /** Return the current user (assertion) level. */
     842                 :        427 :   size_t current_user_level() const { return d_active_vars_control.size(); }
     843                 :            : 
     844                 :            :   /** Return the current list of activation literals. */
     845                 :         33 :   const std::vector<SatLiteral>& activation_literals()
     846                 :            :   {
     847                 :         33 :     return d_activation_literals;
     848                 :            :   }
     849                 :            : 
     850                 :            :   /**
     851                 :            :    * Renotify fixed literals in the current user level.
     852                 :            :    *
     853                 :            :    * This is done prior to a new solve() call and ensures that fixed literals
     854                 :            :    * are enqueued in the theory proxy. This is needed since the SAT solver does
     855                 :            :    * not re-notify us about fixed literals.
     856                 :            :    */
     857                 :         33 :   void renotify_fixed()
     858                 :            :   {
     859                 :         33 :     ++d_stats.renotifyFixed;
     860         [ -  + ]:         33 :     for (const auto& lit : d_renotify_fixed)
     861                 :            :     {
     862         [ -  - ]:          0 :       Trace("cadical::propagator")
     863                 :          0 :           << "re-enqueue (user pop): " << lit << std::endl;
     864                 :            :       // Make sure to pre-register the re-enqueued theory literal
     865                 :          0 :       d_proxy->notifySatLiteral(d_proxy->getNode(lit));
     866                 :            :       // Re-enqueue fixed theory literal
     867                 :          0 :       d_proxy->enqueueTheoryLiteral(lit);
     868                 :            :       // We are notifying fixed literals at the current user level, update the
     869                 :            :       // level at which the variable was fixed, so that it will be renotified,
     870                 :            :       // if needed in lower user levels.
     871                 :          0 :       d_var_info[lit.getSatVariable()].level_fixed = current_user_level();
     872                 :          0 :       ++d_stats.renotifyFixedLit;
     873                 :            :     }
     874                 :         33 :     d_renotify_fixed.clear();
     875                 :         33 :   }
     876                 :            : 
     877                 :            :   /** Set d_in_search flag to indicate whether solver is currently in search. */
     878                 :         66 :   void in_search(bool flag) { d_in_search = flag; }
     879                 :            : 
     880                 :            :  private:
     881                 :            :   /** Retrieve theory propagations and add them to the propagations list. */
     882                 :        160 :   void theory_propagate()
     883                 :            :   {
     884                 :        320 :     SatClause propagated_lits;
     885                 :        160 :     d_proxy->theoryPropagate(propagated_lits);
     886         [ +  - ]:        320 :     Trace("cadical::propagator")
     887                 :        160 :         << "new propagations: " << propagated_lits.size() << std::endl;
     888                 :            : 
     889         [ +  + ]:        221 :     for (const auto& lit : propagated_lits)
     890                 :            :     {
     891         [ +  - ]:         61 :       Trace("cadical::propagator") << "new propagation: " << lit << std::endl;
     892                 :         61 :       d_propagations.push_back(lit);
     893                 :            :     }
     894                 :        160 :   }
     895                 :            : 
     896                 :            :   /**
     897                 :            :    * Get next propagation.
     898                 :            :    *
     899                 :            :    * @return Return next propagation queued in d_propagations.
     900                 :            :    */
     901                 :        130 :   int next_propagation()
     902                 :            :   {
     903         [ +  + ]:        130 :     if (d_propagations.empty())
     904                 :            :     {
     905                 :         79 :       return 0;
     906                 :            :     }
     907                 :         51 :     SatLiteral next = d_propagations.front();
     908                 :         51 :     d_propagations.pop_front();
     909         [ +  - ]:         51 :     Trace("cadical::propagator") << "propagate: " << next << std::endl;
     910                 :         51 :     return toCadicalLit(next);
     911                 :            :   }
     912                 :            : 
     913                 :            :   /** The associated theory proxy. */
     914                 :            :   prop::TheoryProxy* d_proxy = nullptr;
     915                 :            : 
     916                 :            :   /** The SAT context. */
     917                 :            :   context::Context& d_context;
     918                 :            :   CaDiCaL::Solver& d_solver;
     919                 :            : 
     920                 :            :   /** Struct to store information on variables. */
     921                 :            :   struct VarInfo
     922                 :            :   {
     923                 :            :     uint32_t level_intro = 0;     // user level at which variable was created
     924                 :            :     uint32_t level_fixed = 0;     // user level at which variable was fixed
     925                 :            :     bool is_theory_atom = false;  // is variable a theory atom
     926                 :            :     bool is_fixed = false;        // has variable fixed assignment
     927                 :            :     bool is_active = true;        // is variable active
     928                 :            :     int32_t assignment = 0;       // current variable assignment
     929                 :            :     int8_t phase = 0;             // preferred phase
     930                 :            :   };
     931                 :            :   /** Maps SatVariable to corresponding info struct. */
     932                 :            :   std::vector<VarInfo> d_var_info;
     933                 :            : 
     934                 :            :   /**
     935                 :            :    * Currently active variables, can get inactive on user pops.
     936                 :            :    * Dependent on user context.
     937                 :            :    */
     938                 :            :   std::vector<SatVariable> d_active_vars;
     939                 :            :   /**
     940                 :            :    * Control stack to mananage d_active_vars on user pop.
     941                 :            :    *
     942                 :            :    * Note: We do not use a User-context-dependent CDList here, since we neeed
     943                 :            :    *       to know which variables are popped and thus become inactive.
     944                 :            :    */
     945                 :            :   std::vector<size_t> d_active_vars_control;
     946                 :            : 
     947                 :            :   /**
     948                 :            :    * Current activation literals.
     949                 :            :    *
     950                 :            :    * For each user level, we push a fresh activation literal to the vector (in
     951                 :            :    * user_pop()). Activation literals get removed and disabled in user_pop().
     952                 :            :    * The size of the vector corresponds to the current user level.
     953                 :            :    *
     954                 :            :    * The activation literals corrsponding to the current user level gets
     955                 :            :    * automtically added to each clause added in this user level. With
     956                 :            :    * activation literals we can simulate push/pop of clauses in the SAT solver.
     957                 :            :    */
     958                 :            :   std::vector<SatLiteral> d_activation_literals;
     959                 :            : 
     960                 :            :   /** List of fixed literals to be re-notified in lower user level. */
     961                 :            :   std::vector<SatLiteral> d_renotify_fixed;
     962                 :            : 
     963                 :            :   /**
     964                 :            :    * Variable assignment notifications.
     965                 :            :    *
     966                 :            :    * Used to unassign variables on backtrack.
     967                 :            :    */
     968                 :            :   std::vector<SatLiteral> d_assignments;
     969                 :            :   /**
     970                 :            :    * Control stack to manage d_assignments when backtracking on SAT level.
     971                 :            :    *
     972                 :            :    * Note: We do not use a SAT-context-depenent CDList for d_assignments, since
     973                 :            :    *       we need to know which non-fixed variables are unassigned on
     974                 :            :    *       backtrack.
     975                 :            :    */
     976                 :            :   std::vector<size_t> d_assignment_control;
     977                 :            : 
     978                 :            :   /**
     979                 :            :    * Stores all observed decision variables.
     980                 :            :    *
     981                 :            :    * Note: May contain undefSatLiteral for unobserved decision variables.
     982                 :            :    */
     983                 :            :   std::vector<SatLiteral> d_decisions;
     984                 :            : 
     985                 :            :   /** Used by cb_propagate() to return propagated literals. */
     986                 :            :   std::deque<SatLiteral> d_propagations;
     987                 :            : 
     988                 :            :   /**
     989                 :            :    * Used by add_clause() to buffer added clauses, which will be added via
     990                 :            :    * cb_add_reason_clause_lit().
     991                 :            :    */
     992                 :            :   std::deque<CadicalLit> d_new_clauses;
     993                 :            : 
     994                 :            :   /**
     995                 :            :    * Flag indicating whether cb_add_reason_clause_lit() is currently
     996                 :            :    * processing a reason.
     997                 :            :    */
     998                 :            :   bool d_processing_reason = false;
     999                 :            :   /** Reason storage to process current reason in cb_add_reason_clause_lit(). */
    1000                 :            :   std::deque<SatLiteral> d_reason;
    1001                 :            : 
    1002                 :            :   bool d_found_solution = false;
    1003                 :            : 
    1004                 :            :   /** Flag indicating if SAT solver is in search(). */
    1005                 :            :   bool d_in_search = false;
    1006                 :            : 
    1007                 :            :   struct Statistics
    1008                 :            :   {
    1009                 :         15 :     Statistics(StatisticsRegistry& stats)
    1010                 :         15 :         : renotifyFixed(
    1011                 :         15 :               stats.registerInt("cadical::propagator::renotify_fixed")),
    1012                 :            :           renotifyFixedLit(
    1013                 :         15 :               stats.registerInt("cadical::propagator::renotify_fixed_lit")),
    1014                 :            :           notifyAssignment(
    1015                 :         15 :               stats.registerInt("cadical::propagator::notify_assignment")),
    1016                 :            :           notifyFixedAssignment(stats.registerInt(
    1017                 :         15 :               "cadical::propagator::notify_fixed_assignment")),
    1018                 :            :           notifyNewDecision(
    1019                 :         15 :               stats.registerInt("cadical::propagator::notify_new_decision")),
    1020                 :            :           notifyBacktrack(
    1021                 :         15 :               stats.registerInt("cadical::propagator::notify_backtrack")),
    1022                 :            :           cbCheckFoundModel(
    1023                 :         15 :               stats.registerInt("cadical::propagator::cb_check_found_model")),
    1024                 :         15 :           cbDecide(stats.registerInt("cadical::propagator::cb_decide")),
    1025                 :         15 :           cbPropagate(stats.registerInt("cadical::propagator::cb_propagate")),
    1026                 :            :           cbAddReasonClauseLit(stats.registerInt(
    1027                 :         15 :               "cadical::propagator::cb_add_reason_clause_lit")),
    1028                 :            :           cbHasExternalClause(
    1029                 :         15 :               stats.registerInt("cadical::propagator::cb_has_external_clause")),
    1030                 :            :           cbAddExternalClauseLit(stats.registerInt(
    1031                 :         15 :               "cadical::propagator::cb_add_external_clause_lit"))
    1032                 :            :     {
    1033                 :         15 :     }
    1034                 :            :     IntStat renotifyFixed;
    1035                 :            :     IntStat renotifyFixedLit;
    1036                 :            :     IntStat notifyAssignment;
    1037                 :            :     IntStat notifyFixedAssignment;
    1038                 :            :     IntStat notifyNewDecision;
    1039                 :            :     IntStat notifyBacktrack;
    1040                 :            :     IntStat cbCheckFoundModel;
    1041                 :            :     IntStat cbDecide;
    1042                 :            :     IntStat cbPropagate;
    1043                 :            :     IntStat cbAddReasonClauseLit;
    1044                 :            :     IntStat cbHasExternalClause;
    1045                 :            :     IntStat cbAddExternalClauseLit;
    1046                 :            :   } d_stats;
    1047                 :            : };
    1048                 :            : 
    1049                 :            : class ClauseLearner : public CaDiCaL::Learner
    1050                 :            : {
    1051                 :            :  public:
    1052                 :          1 :   ClauseLearner(TheoryProxy& proxy, int32_t clause_size)
    1053                 :          1 :       : d_proxy(proxy), d_max_clause_size(clause_size)
    1054                 :            :   {
    1055                 :          1 :   }
    1056                 :          2 :   ~ClauseLearner() override {}
    1057                 :            : 
    1058                 :          1 :   bool learning(int size) override
    1059                 :            :   {
    1060 [ -  + ][ -  - ]:          1 :     return d_max_clause_size == 0 || size <= d_max_clause_size;
    1061                 :            :   }
    1062                 :            : 
    1063                 :          2 :   void learn(int lit) override
    1064                 :            :   {
    1065         [ +  + ]:          2 :     if (lit)
    1066                 :            :     {
    1067                 :          1 :       SatLiteral slit = toSatLiteral(lit);
    1068                 :          1 :       d_clause.push_back(slit);
    1069                 :            :     }
    1070                 :            :     else
    1071                 :            :     {
    1072                 :          1 :       d_proxy.notifySatClause(d_clause);
    1073                 :          1 :       d_clause.clear();
    1074                 :            :     }
    1075                 :          2 :   }
    1076                 :            : 
    1077                 :            :  private:
    1078                 :            :   TheoryProxy& d_proxy;
    1079                 :            :   /** Intermediate literals buffer. */
    1080                 :            :   std::vector<SatLiteral> d_clause;
    1081                 :            :   /** Maximum size of clauses to get notified about. */
    1082                 :            :   int32_t d_max_clause_size;
    1083                 :            : };
    1084                 :            : 
    1085                 :      40099 : CadicalSolver::CadicalSolver(Env& env,
    1086                 :            :                              StatisticsRegistry& registry,
    1087                 :      40099 :                              const std::string& name)
    1088                 :            :     : EnvObj(env),
    1089                 :      40099 :       d_solver(new CaDiCaL::Solver()),
    1090                 :      40099 :       d_context(context()),
    1091                 :            :       // Note: CaDiCaL variables start with index 1 rather than 0 since negated
    1092                 :            :       //       literals are represented as the negation of the index.
    1093                 :            :       d_nextVarIdx(1),
    1094                 :            :       d_inSatMode(false),
    1095                 :     120297 :       d_statistics(registry, name)
    1096                 :            : {
    1097                 :      40099 : }
    1098                 :            : 
    1099                 :      40099 : void CadicalSolver::init()
    1100                 :            : {
    1101                 :      40099 :   d_solver->set("quiet", 1);  // CaDiCaL is verbose by default
    1102                 :            : 
    1103                 :            :   // walk and lucky phase do not use the external propagator, disable for now
    1104         [ +  + ]:      40099 :   if (d_propagator)
    1105                 :            :   {
    1106                 :         15 :     d_solver->set("walk", 0);
    1107                 :         15 :     d_solver->set("lucky", 0);
    1108                 :            :     // ilb currently does not play well with user propagators
    1109                 :         15 :     d_solver->set("ilb", 0);
    1110                 :         15 :     d_solver->set("ilbassumptions", 0);
    1111         [ +  - ]:         15 :     d_solver->connect_fixed_listener(d_propagator.get());
    1112                 :         15 :     d_solver->connect_external_propagator(d_propagator.get());
    1113                 :            :   }
    1114                 :            : 
    1115                 :      40099 :   d_true = newVar();
    1116                 :      40099 :   d_false = newVar();
    1117                 :      40099 :   d_solver->add(toCadicalVar(d_true));
    1118                 :      40099 :   d_solver->add(0);
    1119                 :      40099 :   d_solver->add(-toCadicalVar(d_false));
    1120                 :      40099 :   d_solver->add(0);
    1121                 :            : 
    1122                 :      40099 :   bool logProofs = false;
    1123                 :            :   // TODO (wishue #154): determine how to initialize the proofs for CaDiCaL
    1124                 :            :   // here based on d_env.isSatProofProducing and options().proof.propProofMode.
    1125                 :            :   // The latter should be extended to include modes DRAT and LRAT based on
    1126                 :            :   // what is available here.
    1127         [ -  + ]:      40099 :   if (logProofs)
    1128                 :            :   {
    1129                 :          0 :     d_pfFile = options().driver.filename + ".drat_proof.txt";
    1130         [ -  - ]:          0 :     if (!options().proof.dratBinaryFormat)
    1131                 :            :     {
    1132                 :          0 :       d_solver->set("binary", 0);
    1133                 :            :     }
    1134                 :          0 :     d_solver->set("inprocessing", 0);
    1135                 :          0 :     d_solver->trace_proof(d_pfFile.c_str());
    1136                 :            :   }
    1137                 :      40099 : }
    1138                 :            : 
    1139                 :      79702 : CadicalSolver::~CadicalSolver() {}
    1140                 :            : 
    1141                 :            : /**
    1142                 :            :  * Terminator class that notifies CaDiCaL to terminate when the resource limit
    1143                 :            :  * is reached (used for resource limits specified via --rlimit or --tlimit).
    1144                 :            :  */
    1145                 :            : class ResourceLimitTerminator : public CaDiCaL::Terminator
    1146                 :            : {
    1147                 :            :  public:
    1148                 :      40099 :   ResourceLimitTerminator(ResourceManager& resmgr) : d_resmgr(resmgr){};
    1149                 :            : 
    1150                 :     829916 :   bool terminate() override
    1151                 :            :   {
    1152                 :     829916 :     d_resmgr.spendResource(Resource::BvSatStep);
    1153                 :     829916 :     return d_resmgr.out();
    1154                 :            :   }
    1155                 :            : 
    1156                 :            :  private:
    1157                 :            :   ResourceManager& d_resmgr;
    1158                 :            : };
    1159                 :            : 
    1160                 :      40099 : void CadicalSolver::setResourceLimit(ResourceManager* resmgr)
    1161                 :            : {
    1162                 :      40099 :   d_terminator.reset(new ResourceLimitTerminator(*resmgr));
    1163                 :      40099 :   d_solver->connect_terminator(d_terminator.get());
    1164                 :      40099 : }
    1165                 :            : 
    1166                 :      68903 : SatValue CadicalSolver::_solve(const std::vector<SatLiteral>& assumptions)
    1167                 :            : {
    1168         [ +  + ]:      68903 :   if (d_propagator)
    1169                 :            :   {
    1170         [ +  - ]:         33 :     Trace("cadical::propagator") << "solve start" << std::endl;
    1171                 :         33 :     d_propagator->renotify_fixed();
    1172                 :            :   }
    1173                 :      68903 :   TimerStat::CodeTimer codeTimer(d_statistics.d_solveTime);
    1174                 :      68903 :   d_assumptions.clear();
    1175         [ +  + ]:      68903 :   if (d_propagator)
    1176                 :            :   {
    1177                 :            :     // Assume activation literals for all active user levels.
    1178         [ +  + ]:         53 :     for (const auto& lit : d_propagator->activation_literals())
    1179                 :            :     {
    1180         [ +  - ]:         40 :       Trace("cadical::propagator")
    1181                 :         20 :           << "assume activation lit: " << ~lit << std::endl;
    1182                 :         20 :       d_solver->assume(toCadicalLit(~lit));
    1183                 :            :     }
    1184                 :            :   }
    1185                 :            :   SatValue res;
    1186         [ +  + ]:    7767900 :   for (const SatLiteral& lit : assumptions)
    1187                 :            :   {
    1188         [ -  + ]:    7699000 :     if (d_propagator)
    1189                 :            :     {
    1190         [ -  - ]:          0 :       Trace("cadical::propagator") << "assume: " << lit << std::endl;
    1191                 :            :     }
    1192                 :    7699000 :     d_solver->assume(toCadicalLit(lit));
    1193                 :    7699000 :     d_assumptions.push_back(lit);
    1194                 :            :   }
    1195         [ +  + ]:      68903 :   if (d_propagator)
    1196                 :            :   {
    1197                 :         33 :     d_propagator->in_search(true);
    1198                 :            :   }
    1199                 :      68903 :   res = toSatValue(d_solver->solve());
    1200         [ +  + ]:      68903 :   if (d_propagator)
    1201                 :            :   {
    1202 [ +  + ][ -  + ]:         33 :     Assert(res != SAT_VALUE_TRUE || d_propagator->done());
         [ -  + ][ -  - ]
    1203         [ +  - ]:         33 :     Trace("cadical::propagator") << "solve done: " << res << std::endl;
    1204                 :         33 :     d_propagator->in_search(false);
    1205                 :            :   }
    1206                 :      68903 :   ++d_statistics.d_numSatCalls;
    1207                 :      68903 :   d_inSatMode = (res == SAT_VALUE_TRUE);
    1208                 :     137806 :   return res;
    1209                 :            : }
    1210                 :            : 
    1211                 :            : /* SatSolver Interface ------------------------------------------------------ */
    1212                 :            : 
    1213                 :    5588920 : ClauseId CadicalSolver::addClause(SatClause& clause, bool removable)
    1214                 :            : {
    1215                 :    5588920 :   if (d_propagator && TraceIsOn("cadical::propagator"))
    1216                 :            :   {
    1217         [ -  - ]:          0 :     Trace("cadical::propagator") << "addClause (" << removable << "):";
    1218                 :          0 :     SatLiteral alit = d_propagator->current_activation_lit();
    1219         [ -  - ]:          0 :     if (alit != undefSatLiteral)
    1220                 :            :     {
    1221         [ -  - ]:          0 :       Trace("cadical::propagator") << " " << alit;
    1222                 :            :     }
    1223         [ -  - ]:          0 :     for (const SatLiteral& lit : clause)
    1224                 :            :     {
    1225         [ -  - ]:          0 :       Trace("cadical::propagator") << " " << lit;
    1226                 :            :     }
    1227         [ -  - ]:          0 :     Trace("cadical::propagator") << " 0" << std::endl;
    1228                 :            :   }
    1229                 :            :   // If we are currently in search, add clauses through the propagator.
    1230         [ +  + ]:    5588920 :   if (d_propagator)
    1231                 :            :   {
    1232                 :        207 :     d_propagator->add_clause(clause);
    1233                 :            :   }
    1234                 :            :   else
    1235                 :            :   {
    1236         [ +  + ]:   20807100 :     for (const SatLiteral& lit : clause)
    1237                 :            :     {
    1238                 :   15218400 :       d_solver->add(toCadicalLit(lit));
    1239                 :            :     }
    1240                 :    5588720 :     d_solver->add(0);
    1241                 :            :   }
    1242                 :    5588920 :   ++d_statistics.d_numClauses;
    1243                 :    5588920 :   return ClauseIdError;
    1244                 :            : }
    1245                 :            : 
    1246                 :          0 : ClauseId CadicalSolver::addXorClause(SatClause& clause,
    1247                 :            :                                      bool rhs,
    1248                 :            :                                      bool removable)
    1249                 :            : {
    1250                 :          0 :   Unreachable() << "CaDiCaL does not support adding XOR clauses.";
    1251                 :            :   return 0;
    1252                 :            : }
    1253                 :            : 
    1254                 :    1507890 : SatVariable CadicalSolver::newVar(bool isTheoryAtom, bool canErase)
    1255                 :            : {
    1256                 :    1507890 :   ++d_statistics.d_numVariables;
    1257         [ +  + ]:    1507890 :   if (d_propagator)
    1258                 :            :   {
    1259                 :        150 :     d_propagator->add_new_var(d_nextVarIdx, isTheoryAtom);
    1260                 :            :   }
    1261                 :    1507890 :   return d_nextVarIdx++;
    1262                 :            : }
    1263                 :            : 
    1264                 :        111 : SatVariable CadicalSolver::trueVar() { return d_true; }
    1265                 :            : 
    1266                 :        237 : SatVariable CadicalSolver::falseVar() { return d_false; }
    1267                 :            : 
    1268                 :         33 : SatValue CadicalSolver::solve() { return _solve({}); }
    1269                 :            : 
    1270                 :          0 : SatValue CadicalSolver::solve(long unsigned int&)
    1271                 :            : {
    1272                 :          0 :   Unimplemented() << "Setting limits for CaDiCaL not supported yet";
    1273                 :            :   return SatValue::SAT_VALUE_UNKNOWN;
    1274                 :            : };
    1275                 :            : 
    1276                 :      68870 : SatValue CadicalSolver::solve(const std::vector<SatLiteral>& assumptions)
    1277                 :            : {
    1278                 :      68870 :   return _solve(assumptions);
    1279                 :            : }
    1280                 :            : 
    1281                 :          0 : bool CadicalSolver::setPropagateOnly()
    1282                 :            : {
    1283                 :          0 :   d_solver->limit("decisions", 0); /* Gets reset after next solve() call. */
    1284                 :          0 :   return true;
    1285                 :            : }
    1286                 :            : 
    1287                 :      20076 : void CadicalSolver::getUnsatAssumptions(std::vector<SatLiteral>& assumptions)
    1288                 :            : {
    1289         [ +  + ]:    7513920 :   for (const SatLiteral& lit : d_assumptions)
    1290                 :            :   {
    1291         [ +  + ]:    7493840 :     if (d_solver->failed(toCadicalLit(lit)))
    1292                 :            :     {
    1293                 :     153603 :       assumptions.push_back(lit);
    1294                 :            :     }
    1295                 :            :   }
    1296                 :      20076 : }
    1297                 :            : 
    1298                 :          0 : void CadicalSolver::interrupt() { d_solver->terminate(); }
    1299                 :            : 
    1300                 :        401 : SatValue CadicalSolver::value(SatLiteral l) { return d_propagator->value(l); }
    1301                 :            : 
    1302                 :     552886 : SatValue CadicalSolver::modelValue(SatLiteral l)
    1303                 :            : {
    1304 [ -  + ][ -  + ]:     552886 :   Assert(d_inSatMode);
                 [ -  - ]
    1305                 :     552886 :   auto val = d_solver->val(toCadicalLit(l.getSatVariable()));
    1306         [ +  + ]:     552886 :   return toSatValueLit(l.isNegated() ? -val : val);
    1307                 :            : }
    1308                 :            : 
    1309                 :         15 : uint32_t CadicalSolver::getAssertionLevel() const
    1310                 :            : {
    1311 [ -  + ][ -  + ]:         15 :   Assert(d_propagator);
                 [ -  - ]
    1312                 :         15 :   return d_propagator->current_user_level();
    1313                 :            : }
    1314                 :            : 
    1315                 :          0 : bool CadicalSolver::ok() const { return d_inSatMode; }
    1316                 :            : 
    1317                 :      40099 : CadicalSolver::Statistics::Statistics(StatisticsRegistry& registry,
    1318                 :      40099 :                                       const std::string& prefix)
    1319                 :      40099 :     : d_numSatCalls(registry.registerInt(prefix + "cadical::calls_to_solve")),
    1320                 :      40099 :       d_numVariables(registry.registerInt(prefix + "cadical::variables")),
    1321                 :      40099 :       d_numClauses(registry.registerInt(prefix + "cadical::clauses")),
    1322                 :      40099 :       d_solveTime(registry.registerTimer(prefix + "cadical::solve_time"))
    1323                 :            :   {
    1324                 :      40099 : }
    1325                 :            : 
    1326                 :            : /* CDCLTSatSolver Interface ------------------------------------------------- */
    1327                 :            : 
    1328                 :         15 : void CadicalSolver::initialize(prop::TheoryProxy* theoryProxy,
    1329                 :            :                                PropPfManager* ppm)
    1330                 :            : {
    1331                 :         15 :   d_proxy = theoryProxy;
    1332                 :         15 :   d_propagator.reset(new CadicalPropagator(
    1333                 :         15 :       theoryProxy, d_context, *d_solver, statisticsRegistry()));
    1334         [ +  + ]:         15 :   if (!d_env.getPlugins().empty())
    1335                 :            :   {
    1336                 :          1 :     d_clause_learner.reset(new ClauseLearner(*theoryProxy, 0));
    1337                 :          1 :     d_solver->connect_learner(d_clause_learner.get());
    1338                 :            :   }
    1339                 :            : 
    1340                 :         15 :   init();
    1341                 :         15 : }
    1342                 :            : 
    1343                 :         20 : void CadicalSolver::push()
    1344                 :            : {
    1345                 :         20 :   d_context->push();  // SAT context for cvc5
    1346                 :            :   // Push new user level
    1347                 :         20 :   d_propagator->user_push();
    1348                 :            :   // Set new activation literal for pushed user level
    1349                 :            :   // Note: This happens after the push to ensure that the activation literal's
    1350                 :            :   // introduction level is the current user level.
    1351                 :         20 :   SatVariable alit = newVar(false);
    1352                 :         20 :   d_propagator->set_activation_lit(alit);
    1353                 :         20 : }
    1354                 :            : 
    1355                 :         20 : void CadicalSolver::pop()
    1356                 :            : {
    1357                 :         20 :   d_context->pop();  // SAT context for cvc5
    1358                 :         20 :   d_propagator->user_pop();
    1359                 :            :   // CaDiCaL issues notify_backtrack(0) when done, we don't have to call this
    1360                 :            :   // explicitly here
    1361                 :         20 : }
    1362                 :            : 
    1363                 :         33 : void CadicalSolver::resetTrail()
    1364                 :            : {
    1365                 :            :   // Reset SAT context to decision level 0
    1366                 :         33 :   d_propagator->notify_backtrack(0);
    1367                 :         33 : }
    1368                 :            : 
    1369                 :          2 : void CadicalSolver::preferPhase(SatLiteral lit)
    1370                 :            : {
    1371         [ +  - ]:          2 :   Trace("cadical::propagator") << "phase: " << lit << std::endl;
    1372                 :          2 :   d_propagator->phase(lit);
    1373                 :          2 : }
    1374                 :            : 
    1375                 :          2 : bool CadicalSolver::isDecision(SatVariable var) const
    1376                 :            : {
    1377                 :          2 :   return d_solver->is_decision(toCadicalVar(var));
    1378                 :            : }
    1379                 :            : 
    1380                 :          0 : bool CadicalSolver::isFixed(SatVariable var) const
    1381                 :            : {
    1382         [ -  - ]:          0 :   if (d_propagator)
    1383                 :            :   {
    1384                 :          0 :     return d_propagator->is_fixed(var);
    1385                 :            :   }
    1386                 :          0 :   return d_solver->fixed(toCadicalVar(var));
    1387                 :            : }
    1388                 :            : 
    1389                 :          0 : std::vector<SatLiteral> CadicalSolver::getDecisions() const
    1390                 :            : {
    1391                 :          0 :   std::vector<SatLiteral> decisions;
    1392         [ -  - ]:          0 :   for (SatLiteral lit : d_propagator->get_decisions())
    1393                 :            :   {
    1394         [ -  - ]:          0 :     if (lit != 0)
    1395                 :            :     {
    1396                 :          0 :       decisions.push_back(lit);
    1397                 :            :     }
    1398                 :            :   }
    1399                 :          0 :   return decisions;
    1400                 :            : }
    1401                 :            : 
    1402                 :          0 : std::vector<Node> CadicalSolver::getOrderHeap() const { return {}; }
    1403                 :            : 
    1404                 :          0 : std::shared_ptr<ProofNode> CadicalSolver::getProof()
    1405                 :            : {
    1406                 :            :   // NOTE: we could return a DRAT_REFUTATION or LRAT_REFUTATION proof node
    1407                 :            :   // consisting of a single step, referencing the files for the DIMACS + proof.
    1408                 :            :   // do not throw an exception, since we test whether the proof is available
    1409                 :            :   // by comparing it to nullptr.
    1410                 :          0 :   return nullptr;
    1411                 :            : }
    1412                 :            : 
    1413                 :            : /* -------------------------------------------------------------------------- */
    1414                 :            : }  // namespace prop
    1415                 :            : }  // namespace cvc5::internal

Generated by: LCOV version 1.14