LCOV - code coverage report
Current view: top level - buildbot/coverage/build/src/prop - cnf_stream.cpp (source / functions) Hit Total Coverage
Test: coverage.info Lines: 399 435 91.7 %
Date: 2025-03-28 11:57:18 Functions: 31 34 91.2 %
Branches: 207 447 46.3 %

           Branch data     Line data    Source code
       1                 :            : /******************************************************************************
       2                 :            :  * Top contributors (to current version):
       3                 :            :  *   Dejan Jovanovic, Haniel Barbosa, 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                 :            :  * A CNF converter that takes in asserts and has the side effect of given an
      14                 :            :  * equisatisfiable stream of assertions to PropEngine.
      15                 :            :  */
      16                 :            : #include "prop/cnf_stream.h"
      17                 :            : 
      18                 :            : #include <queue>
      19                 :            : 
      20                 :            : #include "base/check.h"
      21                 :            : #include "base/output.h"
      22                 :            : #include "expr/node.h"
      23                 :            : #include "options/bv_options.h"
      24                 :            : #include "printer/printer.h"
      25                 :            : #include "proof/clause_id.h"
      26                 :            : #include "prop/minisat/minisat.h"
      27                 :            : #include "prop/prop_engine.h"
      28                 :            : #include "prop/theory_proxy.h"
      29                 :            : #include "smt/env.h"
      30                 :            : #include "theory/theory.h"
      31                 :            : #include "theory/theory_engine.h"
      32                 :            : 
      33                 :            : namespace cvc5::internal {
      34                 :            : namespace prop {
      35                 :            : 
      36                 :      92096 : CnfStream::CnfStream(Env& env,
      37                 :            :                      SatSolver* satSolver,
      38                 :            :                      Registrar* registrar,
      39                 :            :                      context::Context* c,
      40                 :            :                      FormulaLitPolicy flpol,
      41                 :      92096 :                      std::string name)
      42                 :            :     : EnvObj(env),
      43                 :            :       d_satSolver(satSolver),
      44                 :            :       d_booleanVariables(c),
      45                 :            :       d_notifyFormulas(c),
      46                 :            :       d_nodeToLiteralMap(c),
      47                 :            :       d_literalToNodeMap(c),
      48                 :            :       d_flitPolicy(flpol),
      49                 :            :       d_registrar(registrar),
      50                 :            :       d_name(name),
      51                 :            :       d_removable(false),
      52                 :      92096 :       d_stats(statisticsRegistry(), name)
      53                 :            : {
      54                 :      92096 : }
      55                 :            : 
      56                 :   12141700 : bool CnfStream::assertClause(TNode node, SatClause& c)
      57                 :            : {
      58         [ +  - ]:   12141700 :   Trace("cnf") << "Inserting into stream " << c << " node = " << node << "\n";
      59                 :            : 
      60                 :   12141700 :   ClauseId clauseId = d_satSolver->addClause(c, d_removable);
      61                 :            : 
      62                 :   12141700 :   return clauseId != ClauseIdUndef;
      63                 :            : }
      64                 :            : 
      65                 :     644257 : bool CnfStream::assertClause(TNode node, SatLiteral a)
      66                 :            : {
      67                 :     644257 :   SatClause clause(1);
      68                 :     644257 :   clause[0] = a;
      69                 :    1288510 :   return assertClause(node, clause);
      70                 :            : }
      71                 :            : 
      72                 :    5502060 : bool CnfStream::assertClause(TNode node, SatLiteral a, SatLiteral b)
      73                 :            : {
      74                 :    5502060 :   SatClause clause(2);
      75                 :    5502060 :   clause[0] = a;
      76                 :    5502060 :   clause[1] = b;
      77                 :   11004100 :   return assertClause(node, clause);
      78                 :            : }
      79                 :            : 
      80                 :    3460560 : bool CnfStream::assertClause(TNode node,
      81                 :            :                              SatLiteral a,
      82                 :            :                              SatLiteral b,
      83                 :            :                              SatLiteral c)
      84                 :            : {
      85                 :    3460560 :   SatClause clause(3);
      86                 :    3460560 :   clause[0] = a;
      87                 :    3460560 :   clause[1] = b;
      88                 :    3460560 :   clause[2] = c;
      89                 :    6921130 :   return assertClause(node, clause);
      90                 :            : }
      91                 :            : 
      92                 :  136736000 : bool CnfStream::hasLiteral(TNode n) const {
      93                 :  136736000 :   NodeToLiteralMap::const_iterator find = d_nodeToLiteralMap.find(n);
      94                 :  136736000 :   return find != d_nodeToLiteralMap.end();
      95                 :            : }
      96                 :            : 
      97                 :    8069530 : void CnfStream::ensureMappingForLiteral(TNode n)
      98                 :            : {
      99                 :    8069530 :   SatLiteral lit = getLiteral(n);
     100         [ +  + ]:    8069530 :   if (!d_literalToNodeMap.contains(lit))
     101                 :            :   {
     102                 :            :     // Store backward-mappings
     103                 :        449 :     d_literalToNodeMap.insert(lit, n);
     104                 :        449 :     d_literalToNodeMap.insert(~lit, n.notNode());
     105                 :            :   }
     106                 :    8069530 : }
     107                 :            : 
     108                 :    7801160 : void CnfStream::ensureLiteral(TNode n)
     109                 :            : {
     110                 :    7801160 :   AlwaysAssertArgument(
     111                 :            :       hasLiteral(n) || n.getType().isBoolean(),
     112                 :            :       n,
     113                 :            :       "ProofCnfStream::ensureLiteral() requires a node of Boolean type.\n"
     114                 :            :       "got node: %s\n"
     115                 :            :       "its type: %s\n",
     116                 :            :       n.toString().c_str(),
     117                 :            :       n.getType().toString().c_str());
     118         [ +  - ]:    7801160 :   Trace("cnf") << "ensureLiteral(" << n << ")\n";
     119                 :    7801160 :   TimerStat::CodeTimer codeTimer(d_stats.d_cnfConversionTime, true);
     120         [ +  + ]:    7801160 :   if (hasLiteral(n))
     121                 :            :   {
     122                 :    7754160 :     ensureMappingForLiteral(n);
     123                 :    7754160 :     return;
     124                 :            :   }
     125                 :            :   // remove top level negation
     126         [ +  + ]:      46993 :   n = n.getKind() == Kind::NOT ? n[0] : n;
     127 [ +  + ][ +  + ]:      46993 :   if (d_env.theoryOf(n) == theory::THEORY_BOOL && !n.isVar())
         [ +  - ][ +  + ]
                 [ -  - ]
     128                 :            :   {
     129                 :            :     // If we were called with something other than a theory atom (or
     130                 :            :     // Boolean variable), we get a SatLiteral that is definitionally
     131                 :            :     // equal to it.
     132                 :            :     // These are not removable and have no proof ID
     133                 :      35139 :     d_removable = false;
     134                 :            : 
     135                 :      35139 :     SatLiteral lit = toCNF(n, false);
     136                 :            : 
     137                 :            :     // Store backward-mappings
     138                 :            :     // These may already exist
     139                 :      35139 :     d_literalToNodeMap.insert_safe(lit, n);
     140                 :      35139 :     d_literalToNodeMap.insert_safe(~lit, n.notNode());
     141                 :            :   }
     142                 :            :   else
     143                 :            :   {
     144                 :            :     // We have a theory atom or variable.
     145                 :      11854 :     convertAtom(n);
     146                 :            :   }
     147                 :            : }
     148                 :            : 
     149                 :    3927190 : SatLiteral CnfStream::newLiteral(TNode node,
     150                 :            :                                  bool isTheoryAtom,
     151                 :            :                                  bool notifyTheory,
     152                 :            :                                  bool canEliminate)
     153                 :            : {
     154         [ +  - ]:    7854370 :   Trace("cnf") << d_name << "::newLiteral(" << node << ", " << isTheoryAtom
     155                 :          0 :                << ")\n"
     156                 :    3927190 :                << push;
     157 [ -  + ][ -  + ]:    3927190 :   Assert(node.getKind() != Kind::NOT);
                 [ -  - ]
     158                 :            : 
     159                 :            :   // if we are tracking formulas, everything is a theory atom
     160 [ +  + ][ -  + ]:    3927190 :   if (!isTheoryAtom && d_flitPolicy == FormulaLitPolicy::TRACK_AND_NOTIFY)
     161                 :            :   {
     162                 :          0 :     isTheoryAtom = true;
     163                 :          0 :     d_notifyFormulas.insert(node);
     164                 :            :   }
     165                 :            : 
     166                 :            :   // Get the literal for this node
     167                 :    3927190 :   SatLiteral lit;
     168         [ +  + ]:    3927190 :   if (!hasLiteral(node))
     169                 :            :   {
     170         [ +  - ]:    3927180 :     Trace("cnf") << d_name << "::newLiteral: node already registered\n";
     171                 :            :     // If no literal, we'll make one
     172         [ +  + ]:    3927180 :     if (node.getKind() == Kind::CONST_BOOLEAN)
     173                 :            :     {
     174         [ +  - ]:     104302 :       Trace("cnf") << d_name << "::newLiteral: boolean const\n";
     175         [ +  + ]:     104302 :       if (node.getConst<bool>())
     176                 :            :       {
     177                 :      52088 :         lit = SatLiteral(d_satSolver->trueVar());
     178                 :            :       }
     179                 :            :       else
     180                 :            :       {
     181                 :      52214 :         lit = SatLiteral(d_satSolver->falseVar());
     182                 :            :       }
     183                 :            :     }
     184                 :            :     else
     185                 :            :     {
     186         [ +  - ]:    3822880 :       Trace("cnf") << d_name << "::newLiteral: new var\n";
     187                 :    3822880 :       lit = SatLiteral(d_satSolver->newVar(isTheoryAtom, canEliminate));
     188                 :    3822880 :       d_stats.d_numAtoms++;
     189                 :            :     }
     190                 :    3927180 :     d_nodeToLiteralMap.insert(node, lit);
     191                 :    3927180 :     d_nodeToLiteralMap.insert(node.notNode(), ~lit);
     192                 :            :   }
     193                 :            :   else
     194                 :            :   {
     195         [ +  - ]:          7 :     Trace("cnf") << d_name << "::newLiteral: node already registered\n";
     196                 :          7 :     lit = getLiteral(node);
     197                 :            :   }
     198                 :            : 
     199                 :            :   // If it's a theory literal, need to store it for back queries
     200 [ +  + ][ +  + ]:    3927190 :   if (isTheoryAtom || d_flitPolicy == FormulaLitPolicy::TRACK
     201         [ -  + ]:    1270180 :       || d_flitPolicy == FormulaLitPolicy::TRACK_AND_NOTIFY_VAR)
     202                 :            :   {
     203                 :    2657000 :     d_literalToNodeMap.insert_safe(lit, node);
     204                 :    2657000 :     d_literalToNodeMap.insert_safe(~lit, node.notNode());
     205                 :            :   }
     206                 :            : 
     207                 :            :   // If a theory literal, we pre-register it
     208         [ +  + ]:    3927190 :   if (notifyTheory)
     209                 :            :   {
     210                 :            :     // In case we are re-entered due to lemmas, save our state
     211                 :    1639870 :     bool backupRemovable = d_removable;
     212                 :    1639880 :     d_registrar->notifySatLiteral(node);
     213                 :    1639850 :     d_removable = backupRemovable;
     214                 :            :   }
     215                 :            :   // Here, you can have it
     216         [ +  - ]:    3927170 :   Trace("cnf") << "newLiteral(" << node << ") => " << lit << "\n" << pop;
     217                 :    3927170 :   return lit;
     218                 :            : }
     219                 :            : 
     220                 :  107965000 : TNode CnfStream::getNode(const SatLiteral& literal)
     221                 :            : {
     222 [ -  + ][ -  + ]:  107965000 :   Assert(d_literalToNodeMap.find(literal) != d_literalToNodeMap.end());
                 [ -  - ]
     223         [ +  - ]:  107965000 :   Trace("cnf") << "getNode(" << literal << ")\n";
     224         [ +  - ]:  215930000 :   Trace("cnf") << "getNode(" << literal << ") => "
     225                 :  107965000 :                << d_literalToNodeMap[literal] << "\n";
     226                 :  107965000 :   return d_literalToNodeMap[literal];
     227                 :            : }
     228                 :            : 
     229                 :    2484740 : const CnfStream::NodeToLiteralMap& CnfStream::getTranslationCache() const
     230                 :            : {
     231                 :    2484740 :   return d_nodeToLiteralMap;
     232                 :            : }
     233                 :            : 
     234                 :   12308500 : const CnfStream::LiteralToNodeMap& CnfStream::getNodeCache() const
     235                 :            : {
     236                 :   12308500 :   return d_literalToNodeMap;
     237                 :            : }
     238                 :            : 
     239                 :      36100 : void CnfStream::getBooleanVariables(std::vector<TNode>& outputVariables) const {
     240                 :      36100 :   outputVariables.insert(outputVariables.end(),
     241                 :            :                          d_booleanVariables.begin(),
     242                 :      72200 :                          d_booleanVariables.end());
     243                 :      36100 : }
     244                 :            : 
     245                 :          0 : bool CnfStream::isNotifyFormula(TNode node) const
     246                 :            : {
     247                 :          0 :   return d_notifyFormulas.find(node) != d_notifyFormulas.end();
     248                 :            : }
     249                 :            : 
     250                 :    1687900 : SatLiteral CnfStream::convertAtom(TNode node)
     251                 :            : {
     252         [ +  - ]:    1687900 :   Trace("cnf") << "convertAtom(" << node << ")\n";
     253                 :            : 
     254 [ -  + ][ -  + ]:    1687900 :   Assert(!hasLiteral(node)) << "atom already mapped!";
                 [ -  - ]
     255                 :            : 
     256                 :    1687900 :   bool theoryLiteral = false;
     257                 :    1687900 :   bool canEliminate = true;
     258                 :    1687900 :   bool preRegister = false;
     259                 :            : 
     260                 :            :   // Is this a variable add it to the list. We distinguish whether a Boolean
     261                 :            :   // variable has been marked as a "Boolean term skolem". These variables are
     262                 :            :   // introduced by the term formula removal pass (term_formula_removal.h)
     263                 :            :   // and maintained by Env (smt/env.h). We treat such variables as theory atoms
     264                 :            :   // since they may occur in term positions and thus need to be considered e.g.
     265                 :            :   // for theory combination.
     266                 :    1687900 :   bool isInternalBoolVar = false;
     267         [ +  + ]:    1687900 :   if (node.isVar())
     268                 :            :   {
     269                 :      51551 :     isInternalBoolVar = !d_env.isBooleanTermSkolem(node);
     270                 :            :   }
     271         [ +  + ]:    1687900 :   if (isInternalBoolVar)
     272                 :            :   {
     273                 :      48027 :     d_booleanVariables.push_back(node);
     274                 :            :     // if TRACK_AND_NOTIFY_VAR, we are notified when Boolean variables are
     275                 :            :     // asserted. Thus, they are marked as theory literals.
     276         [ -  + ]:      48027 :     if (d_flitPolicy == FormulaLitPolicy::TRACK_AND_NOTIFY_VAR)
     277                 :            :     {
     278                 :          0 :       theoryLiteral = true;
     279                 :            :     }
     280                 :            :   }
     281                 :            :   else
     282                 :            :   {
     283                 :    1639870 :     theoryLiteral = true;
     284                 :    1639870 :     canEliminate = false;
     285                 :    1639870 :     preRegister = true;
     286                 :            :   }
     287                 :            : 
     288                 :            :   // Make a new literal (variables are not considered theory literals)
     289                 :    1687910 :   SatLiteral lit = newLiteral(node, theoryLiteral, preRegister, canEliminate);
     290                 :            :   // Return the resulting literal
     291                 :    1687880 :   return lit;
     292                 :            : }
     293                 :            : 
     294                 :  128154000 : SatLiteral CnfStream::getLiteral(TNode node) {
     295 [ -  + ][ -  + ]:  128154000 :   Assert(!node.isNull()) << "CnfStream: can't getLiteral() of null node";
                 [ -  - ]
     296                 :            : 
     297 [ -  + ][ -  - ]:  256308000 :   Assert(d_nodeToLiteralMap.contains(node))
     298 [ -  + ][ -  + ]:  128154000 :       << "Literal not in the CNF Cache: " << node << "\n";
                 [ -  - ]
     299                 :            : 
     300                 :  128154000 :   SatLiteral literal = d_nodeToLiteralMap[node];
     301         [ +  - ]:  256308000 :   Trace("cnf") << "CnfStream::getLiteral(" << node << ") => " << literal
     302                 :  128154000 :                << "\n";
     303                 :  128154000 :   return literal;
     304                 :            : }
     305                 :            : 
     306                 :     256816 : void CnfStream::handleXor(TNode xorNode)
     307                 :            : {
     308 [ -  + ][ -  + ]:     256816 :   Assert(!hasLiteral(xorNode)) << "Atom already mapped!";
                 [ -  - ]
     309 [ -  + ][ -  + ]:     256816 :   Assert(xorNode.getKind() == Kind::XOR) << "Expecting an XOR expression!";
                 [ -  - ]
     310 [ -  + ][ -  + ]:     256816 :   Assert(xorNode.getNumChildren() == 2) << "Expecting exactly 2 children!";
                 [ -  - ]
     311 [ -  + ][ -  + ]:     256816 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     312         [ +  - ]:     256816 :   Trace("cnf") << "CnfStream::handleXor(" << xorNode << ")\n";
     313                 :            : 
     314                 :     256816 :   SatLiteral a = getLiteral(xorNode[0]);
     315                 :     256816 :   SatLiteral b = getLiteral(xorNode[1]);
     316                 :            : 
     317                 :     256816 :   SatLiteral xorLit = newLiteral(xorNode);
     318                 :            : 
     319                 :     256816 :   assertClause(xorNode.negate(), a, b, ~xorLit);
     320                 :     256816 :   assertClause(xorNode.negate(), ~a, ~b, ~xorLit);
     321                 :     256816 :   assertClause(xorNode, a, ~b, xorLit);
     322                 :     256816 :   assertClause(xorNode, ~a, b, xorLit);
     323                 :     256816 : }
     324                 :            : 
     325                 :     308395 : void CnfStream::handleOr(TNode orNode)
     326                 :            : {
     327 [ -  + ][ -  + ]:     308395 :   Assert(!hasLiteral(orNode)) << "Atom already mapped!";
                 [ -  - ]
     328 [ -  + ][ -  + ]:     308395 :   Assert(orNode.getKind() == Kind::OR) << "Expecting an OR expression!";
                 [ -  - ]
     329 [ -  + ][ -  + ]:     308395 :   Assert(orNode.getNumChildren() > 1) << "Expecting more then 1 child!";
                 [ -  - ]
     330 [ -  + ][ -  + ]:     308395 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     331         [ +  - ]:     308395 :   Trace("cnf") << "CnfStream::handleOr(" << orNode << ")\n";
     332                 :            : 
     333                 :            :   // Number of children
     334                 :     308395 :   size_t numChildren = orNode.getNumChildren();
     335                 :            : 
     336                 :            :   // Get the literal for this node
     337                 :     308395 :   SatLiteral orLit = newLiteral(orNode);
     338                 :            : 
     339                 :            :   // Transform all the children first
     340                 :     308395 :   SatClause clause(numChildren + 1);
     341         [ +  + ]:    1119250 :   for (size_t i = 0; i < numChildren; ++i)
     342                 :            :   {
     343                 :     810858 :     clause[i] = getLiteral(orNode[i]);
     344                 :            : 
     345                 :            :     // lit <- (a_1 | a_2 | a_3 | ... | a_n)
     346                 :            :     // lit | ~(a_1 | a_2 | a_3 | ... | a_n)
     347                 :            :     // (lit | ~a_1) & (lit | ~a_2) & (lit & ~a_3) & ... & (lit & ~a_n)
     348                 :     810858 :     assertClause(orNode, orLit, ~clause[i]);
     349                 :            :   }
     350                 :            : 
     351                 :            :   // lit -> (a_1 | a_2 | a_3 | ... | a_n)
     352                 :            :   // ~lit | a_1 | a_2 | a_3 | ... | a_n
     353                 :     308395 :   clause[numChildren] = ~orLit;
     354                 :            :   // This needs to go last, as the clause might get modified by the SAT solver
     355                 :     308395 :   assertClause(orNode.negate(), clause);
     356                 :     308395 : }
     357                 :            : 
     358                 :     631167 : void CnfStream::handleAnd(TNode andNode)
     359                 :            : {
     360 [ -  + ][ -  + ]:     631167 :   Assert(!hasLiteral(andNode)) << "Atom already mapped!";
                 [ -  - ]
     361 [ -  + ][ -  + ]:     631167 :   Assert(andNode.getKind() == Kind::AND) << "Expecting an AND expression!";
                 [ -  - ]
     362 [ -  + ][ -  + ]:     631167 :   Assert(andNode.getNumChildren() > 1) << "Expecting more than 1 child!";
                 [ -  - ]
     363 [ -  + ][ -  + ]:     631167 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     364         [ +  - ]:     631167 :   Trace("cnf") << "handleAnd(" << andNode << ")\n";
     365                 :            : 
     366                 :            :   // Number of children
     367                 :     631167 :   size_t numChildren = andNode.getNumChildren();
     368                 :            : 
     369                 :            :   // Get the literal for this node
     370                 :     631167 :   SatLiteral andLit = newLiteral(andNode);
     371                 :            : 
     372                 :            :   // Transform all the children first (remembering the negation)
     373                 :     631167 :   SatClause clause(numChildren + 1);
     374         [ +  + ]:    3485700 :   for (size_t i = 0; i < numChildren; ++i)
     375                 :            :   {
     376                 :    2854530 :     clause[i] = ~getLiteral(andNode[i]);
     377                 :            : 
     378                 :            :     // lit -> (a_1 & a_2 & a_3 & ... & a_n)
     379                 :            :     // ~lit | (a_1 & a_2 & a_3 & ... & a_n)
     380                 :            :     // (~lit | a_1) & (~lit | a_2) & ... & (~lit | a_n)
     381                 :    2854530 :     assertClause(andNode.negate(), ~andLit, ~clause[i]);
     382                 :            :   }
     383                 :            : 
     384                 :            :   // lit <- (a_1 & a_2 & a_3 & ... a_n)
     385                 :            :   // lit | ~(a_1 & a_2 & a_3 & ... & a_n)
     386                 :            :   // lit | ~a_1 | ~a_2 | ~a_3 | ... | ~a_n
     387                 :     631167 :   clause[numChildren] = andLit;
     388                 :            :   // This needs to go last, as the clause might get modified by the SAT solver
     389                 :     631167 :   assertClause(andNode, clause);
     390                 :     631167 : }
     391                 :            : 
     392                 :       2744 : void CnfStream::handleImplies(TNode impliesNode)
     393                 :            : {
     394 [ -  + ][ -  + ]:       2744 :   Assert(!hasLiteral(impliesNode)) << "Atom already mapped!";
                 [ -  - ]
     395 [ -  + ][ -  + ]:       2744 :   Assert(impliesNode.getKind() == Kind::IMPLIES)
                 [ -  - ]
     396                 :          0 :       << "Expecting an IMPLIES expression!";
     397 [ -  + ][ -  + ]:       2744 :   Assert(impliesNode.getNumChildren() == 2) << "Expecting exactly 2 children!";
                 [ -  - ]
     398 [ -  + ][ -  + ]:       2744 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     399         [ +  - ]:       2744 :   Trace("cnf") << "handleImplies(" << impliesNode << ")\n";
     400                 :            : 
     401                 :            :   // Convert the children to cnf
     402                 :       2744 :   SatLiteral a = getLiteral(impliesNode[0]);
     403                 :       2744 :   SatLiteral b = getLiteral(impliesNode[1]);
     404                 :            : 
     405                 :       2744 :   SatLiteral impliesLit = newLiteral(impliesNode);
     406                 :            : 
     407                 :            :   // lit -> (a->b)
     408                 :            :   // ~lit | ~ a | b
     409                 :       2744 :   assertClause(impliesNode.negate(), ~impliesLit, ~a, b);
     410                 :            : 
     411                 :            :   // (a->b) -> lit
     412                 :            :   // ~(~a | b) | lit
     413                 :            :   // (a | l) & (~b | l)
     414                 :       2744 :   assertClause(impliesNode, a, impliesLit);
     415                 :       2744 :   assertClause(impliesNode, ~b, impliesLit);
     416                 :       2744 : }
     417                 :            : 
     418                 :     223275 : void CnfStream::handleIff(TNode iffNode)
     419                 :            : {
     420 [ -  + ][ -  + ]:     223275 :   Assert(!hasLiteral(iffNode)) << "Atom already mapped!";
                 [ -  - ]
     421 [ -  + ][ -  + ]:     223275 :   Assert(iffNode.getKind() == Kind::EQUAL) << "Expecting an EQUAL expression!";
                 [ -  - ]
     422 [ -  + ][ -  + ]:     223275 :   Assert(iffNode.getNumChildren() == 2) << "Expecting exactly 2 children!";
                 [ -  - ]
     423 [ -  + ][ -  + ]:     223275 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     424         [ +  - ]:     223275 :   Trace("cnf") << "handleIff(" << iffNode << ")\n";
     425                 :            : 
     426                 :            :   // Convert the children to CNF
     427                 :     223275 :   SatLiteral a = getLiteral(iffNode[0]);
     428                 :     223275 :   SatLiteral b = getLiteral(iffNode[1]);
     429                 :            : 
     430                 :            :   // Get the now literal
     431                 :     223275 :   SatLiteral iffLit = newLiteral(iffNode);
     432                 :            : 
     433                 :            :   // lit -> ((a-> b) & (b->a))
     434                 :            :   // ~lit | ((~a | b) & (~b | a))
     435                 :            :   // (~a | b | ~lit) & (~b | a | ~lit)
     436                 :     223275 :   assertClause(iffNode.negate(), ~a, b, ~iffLit);
     437                 :     223275 :   assertClause(iffNode.negate(), a, ~b, ~iffLit);
     438                 :            : 
     439                 :            :   // (a<->b) -> lit
     440                 :            :   // ~((a & b) | (~a & ~b)) | lit
     441                 :            :   // (~(a & b)) & (~(~a & ~b)) | lit
     442                 :            :   // ((~a | ~b) & (a | b)) | lit
     443                 :            :   // (~a | ~b | lit) & (a | b | lit)
     444                 :     223275 :   assertClause(iffNode, ~a, ~b, iffLit);
     445                 :     223275 :   assertClause(iffNode, a, b, iffLit);
     446                 :     223275 : }
     447                 :            : 
     448                 :      76860 : void CnfStream::handleIte(TNode iteNode)
     449                 :            : {
     450 [ -  + ][ -  + ]:      76860 :   Assert(!hasLiteral(iteNode)) << "Atom already mapped!";
                 [ -  - ]
     451 [ -  + ][ -  + ]:      76860 :   Assert(iteNode.getKind() == Kind::ITE);
                 [ -  - ]
     452 [ -  + ][ -  + ]:      76860 :   Assert(iteNode.getNumChildren() == 3);
                 [ -  - ]
     453 [ -  + ][ -  + ]:      76860 :   Assert(!d_removable) << "Removable clauses can not contain Boolean structure";
                 [ -  - ]
     454                 :     153720 :   Trace("cnf") << "handleIte(" << iteNode[0] << " " << iteNode[1] << " "
     455 [ -  + ][ -  + ]:      76860 :                << iteNode[2] << ")\n";
                 [ -  - ]
     456                 :            : 
     457                 :      76860 :   SatLiteral condLit = getLiteral(iteNode[0]);
     458                 :      76860 :   SatLiteral thenLit = getLiteral(iteNode[1]);
     459                 :      76860 :   SatLiteral elseLit = getLiteral(iteNode[2]);
     460                 :            : 
     461                 :      76860 :   SatLiteral iteLit = newLiteral(iteNode);
     462                 :            : 
     463                 :            :   // If ITE is true then one of the branches is true and the condition
     464                 :            :   // implies which one
     465                 :            :   // lit -> (ite b t e)
     466                 :            :   // lit -> (t | e) & (b -> t) & (!b -> e)
     467                 :            :   // lit -> (t | e) & (!b | t) & (b | e)
     468                 :            :   // (!lit | t | e) & (!lit | !b | t) & (!lit | b | e)
     469                 :      76860 :   assertClause(iteNode.negate(), ~iteLit, thenLit, elseLit);
     470                 :      76860 :   assertClause(iteNode.negate(), ~iteLit, ~condLit, thenLit);
     471                 :      76860 :   assertClause(iteNode.negate(), ~iteLit, condLit, elseLit);
     472                 :            : 
     473                 :            :   // If ITE is false then one of the branches is false and the condition
     474                 :            :   // implies which one
     475                 :            :   // !lit -> !(ite b t e)
     476                 :            :   // !lit -> (!t | !e) & (b -> !t) & (!b -> !e)
     477                 :            :   // !lit -> (!t | !e) & (!b | !t) & (b | !e)
     478                 :            :   // (lit | !t | !e) & (lit | !b | !t) & (lit | b | !e)
     479                 :      76860 :   assertClause(iteNode, iteLit, ~thenLit, ~elseLit);
     480                 :      76860 :   assertClause(iteNode, iteLit, ~condLit, ~thenLit);
     481                 :      76860 :   assertClause(iteNode, iteLit, condLit, ~elseLit);
     482                 :      76860 : }
     483                 :            : 
     484                 :    2389050 : SatLiteral CnfStream::toCNF(TNode node, bool negated)
     485                 :            : {
     486         [ +  - ]:    4778110 :   Trace("cnf") << "toCNF(" << node
     487         [ -  - ]:    2389050 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     488                 :            : 
     489                 :    4778110 :   TNode cur;
     490                 :    2389050 :   SatLiteral nodeLit;
     491                 :    4778110 :   std::vector<TNode> visit;
     492                 :    4778110 :   std::unordered_map<TNode, bool> cache;
     493                 :            : 
     494                 :    2389050 :   visit.push_back(node);
     495         [ +  + ]:   12230400 :   while (!visit.empty())
     496                 :            :   {
     497                 :    9841390 :     cur = visit.back();
     498 [ -  + ][ -  + ]:    9841390 :     Assert(cur.getType().isBoolean());
                 [ -  - ]
     499                 :            : 
     500         [ +  + ]:    9841390 :     if (hasLiteral(cur))
     501                 :            :     {
     502                 :    5204680 :       visit.pop_back();
     503                 :    7613810 :       continue;
     504                 :            :     }
     505                 :            : 
     506                 :    4636710 :     const auto& it = cache.find(cur);
     507         [ +  + ]:    4636710 :     if (it == cache.end())
     508                 :            :     {
     509                 :    2409130 :       cache.emplace(cur, false);
     510                 :    2409130 :       Kind k = cur.getKind();
     511                 :            :       // Only traverse Boolean nodes
     512 [ +  + ][ +  + ]:    2227580 :       if (k == Kind::NOT || k == Kind::XOR || k == Kind::ITE
     513 [ +  + ][ +  + ]:    1893900 :           || k == Kind::IMPLIES || k == Kind::OR || k == Kind::AND
                 [ +  + ]
     514                 :    4636710 :           || (k == Kind::EQUAL && cur[0].getType().isBoolean()))
     515                 :            :       {
     516                 :            :         // Preserve the order of the recursive version
     517         [ +  + ]:    6724020 :         for (size_t i = 0, size = cur.getNumChildren(); i < size; ++i)
     518                 :            :         {
     519                 :    5043200 :           visit.push_back(cur[size - 1 - i]);
     520                 :            :         }
     521                 :            :       }
     522                 :    2409130 :       continue;
     523                 :            :     }
     524         [ +  - ]:    2227580 :     else if (!it->second)
     525                 :            :     {
     526                 :    2227580 :       it->second = true;
     527                 :    2227580 :       Kind k = cur.getKind();
     528 [ -  + ][ +  + ]:    2227580 :       switch (k)
            [ +  + ][ + ]
     529                 :            :       {
     530                 :          0 :         case Kind::NOT: Assert(hasLiteral(cur[0])); break;
     531                 :     256816 :         case Kind::XOR: handleXor(cur); break;
     532                 :      76860 :         case Kind::ITE: handleIte(cur); break;
     533                 :       2744 :         case Kind::IMPLIES: handleImplies(cur); break;
     534                 :     308395 :         case Kind::OR: handleOr(cur); break;
     535                 :     631167 :         case Kind::AND: handleAnd(cur); break;
     536                 :     951593 :         default:
     537                 :     951593 :           if (k == Kind::EQUAL && cur[0].getType().isBoolean())
     538                 :            :           {
     539                 :     223275 :             handleIff(cur);
     540                 :            :           }
     541                 :            :           else
     542                 :            :           {
     543                 :     728334 :             convertAtom(cur);
     544                 :            :           }
     545                 :     951577 :           break;
     546                 :            :       }
     547                 :            :     }
     548                 :    2227560 :     visit.pop_back();
     549                 :            :   }
     550                 :            : 
     551                 :    2389040 :   nodeLit = getLiteral(node);
     552         [ +  - ]:    4778070 :   Trace("cnf") << "toCNF(): resulting literal: "
     553         [ -  - ]:    2389040 :                << (!negated ? nodeLit : ~nodeLit) << "\n";
     554         [ +  + ]:    4778070 :   return negated ? ~nodeLit : nodeLit;
     555                 :            : }
     556                 :            : 
     557                 :     100208 : void CnfStream::convertAndAssertAnd(TNode node, bool negated)
     558                 :            : {
     559 [ -  + ][ -  + ]:     100208 :   Assert(node.getKind() == Kind::AND);
                 [ -  - ]
     560         [ +  - ]:     200416 :   Trace("cnf") << "CnfStream::convertAndAssertAnd(" << node
     561         [ -  - ]:     100208 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     562         [ +  + ]:     100208 :   if (!negated) {
     563                 :            :     // If the node is a conjunction, we handle each conjunct separately
     564                 :      58426 :     for(TNode::const_iterator conjunct = node.begin(), node_end = node.end();
     565         [ +  + ]:      58426 :         conjunct != node_end; ++conjunct ) {
     566                 :      44810 :       convertAndAssert(*conjunct, false);
     567                 :            :     }
     568                 :            :   } else {
     569                 :            :     // If the node is a disjunction, we construct a clause and assert it
     570                 :      86588 :     int nChildren = node.getNumChildren();
     571                 :      86588 :     SatClause clause(nChildren);
     572                 :      86588 :     TNode::const_iterator disjunct = node.begin();
     573         [ +  + ]:    1247160 :     for(int i = 0; i < nChildren; ++ disjunct, ++ i) {
     574 [ -  + ][ -  + ]:    1160570 :       Assert(disjunct != node.end());
                 [ -  - ]
     575                 :    1160570 :       clause[i] = toCNF(*disjunct, true);
     576                 :            :     }
     577 [ -  + ][ -  + ]:      86588 :     Assert(disjunct == node.end());
                 [ -  - ]
     578                 :      86588 :     assertClause(node.negate(), clause);
     579                 :            :   }
     580                 :     100206 : }
     581                 :            : 
     582                 :     233266 : void CnfStream::convertAndAssertOr(TNode node, bool negated)
     583                 :            : {
     584 [ -  + ][ -  + ]:     233266 :   Assert(node.getKind() == Kind::OR);
                 [ -  - ]
     585         [ +  - ]:     466532 :   Trace("cnf") << "CnfStream::convertAndAssertOr(" << node
     586         [ -  - ]:     233266 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     587         [ +  + ]:     233266 :   if (!negated) {
     588                 :            :     // If the node is a disjunction, we construct a clause and assert it
     589                 :     233052 :     int nChildren = node.getNumChildren();
     590                 :     233052 :     SatClause clause(nChildren);
     591                 :     233052 :     TNode::const_iterator disjunct = node.begin();
     592         [ +  + ]:     894011 :     for(int i = 0; i < nChildren; ++ disjunct, ++ i) {
     593 [ -  + ][ -  + ]:     660959 :       Assert(disjunct != node.end());
                 [ -  - ]
     594                 :     660959 :       clause[i] = toCNF(*disjunct, false);
     595                 :            :     }
     596 [ -  + ][ -  + ]:     233052 :     Assert(disjunct == node.end());
                 [ -  - ]
     597                 :     233052 :     assertClause(node, clause);
     598                 :            :   } else {
     599                 :            :     // If the node is a conjunction, we handle each conjunct separately
     600                 :       1293 :     for(TNode::const_iterator conjunct = node.begin(), node_end = node.end();
     601         [ +  + ]:       1293 :         conjunct != node_end; ++conjunct ) {
     602                 :       1079 :       convertAndAssert(*conjunct, true);
     603                 :            :     }
     604                 :            :   }
     605                 :     233266 : }
     606                 :            : 
     607                 :         70 : void CnfStream::convertAndAssertXor(TNode node, bool negated)
     608                 :            : {
     609 [ -  + ][ -  + ]:         70 :   Assert(node.getKind() == Kind::XOR);
                 [ -  - ]
     610         [ +  - ]:        140 :   Trace("cnf") << "CnfStream::convertAndAssertXor(" << node
     611         [ -  - ]:         70 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     612         [ +  + ]:         70 :   if (!negated) {
     613                 :            :     // p XOR q
     614                 :         66 :     SatLiteral p = toCNF(node[0], false);
     615                 :         66 :     SatLiteral q = toCNF(node[1], false);
     616                 :            :     // Construct the clauses (p => !q) and (!q => p)
     617                 :        132 :     SatClause clause1(2);
     618                 :         66 :     clause1[0] = ~p;
     619                 :         66 :     clause1[1] = ~q;
     620                 :         66 :     assertClause(node, clause1);
     621                 :         66 :     SatClause clause2(2);
     622                 :         66 :     clause2[0] = p;
     623                 :         66 :     clause2[1] = q;
     624                 :         66 :     assertClause(node, clause2);
     625                 :            :   } else {
     626                 :            :     // !(p XOR q) is the same as p <=> q
     627                 :          4 :     SatLiteral p = toCNF(node[0], false);
     628                 :          4 :     SatLiteral q = toCNF(node[1], false);
     629                 :            :     // Construct the clauses (p => q) and (q => p)
     630                 :          8 :     SatClause clause1(2);
     631                 :          4 :     clause1[0] = ~p;
     632                 :          4 :     clause1[1] = q;
     633                 :          4 :     assertClause(node.negate(), clause1);
     634                 :          4 :     SatClause clause2(2);
     635                 :          4 :     clause2[0] = p;
     636                 :          4 :     clause2[1] = ~q;
     637                 :          4 :     assertClause(node.negate(), clause2);
     638                 :            :   }
     639                 :         70 : }
     640                 :            : 
     641                 :       8687 : void CnfStream::convertAndAssertIff(TNode node, bool negated)
     642                 :            : {
     643 [ -  + ][ -  + ]:       8687 :   Assert(node.getKind() == Kind::EQUAL);
                 [ -  - ]
     644         [ +  - ]:      17374 :   Trace("cnf") << "CnfStream::convertAndAssertIff(" << node
     645         [ -  - ]:       8687 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     646         [ +  + ]:       8687 :   if (!negated) {
     647                 :            :     // p <=> q
     648                 :       8191 :     SatLiteral p = toCNF(node[0], false);
     649                 :       8191 :     SatLiteral q = toCNF(node[1], false);
     650                 :            :     // Construct the clauses (p => q) and (q => p)
     651                 :      16382 :     SatClause clause1(2);
     652                 :       8191 :     clause1[0] = ~p;
     653                 :       8191 :     clause1[1] = q;
     654                 :       8191 :     assertClause(node, clause1);
     655                 :       8191 :     SatClause clause2(2);
     656                 :       8191 :     clause2[0] = p;
     657                 :       8191 :     clause2[1] = ~q;
     658                 :       8191 :     assertClause(node, clause2);
     659                 :            :   } else {
     660                 :            :     // !(p <=> q) is the same as p XOR q
     661                 :        496 :     SatLiteral p = toCNF(node[0], false);
     662                 :        496 :     SatLiteral q = toCNF(node[1], false);
     663                 :            :     // Construct the clauses (p => !q) and (!q => p)
     664                 :        992 :     SatClause clause1(2);
     665                 :        496 :     clause1[0] = ~p;
     666                 :        496 :     clause1[1] = ~q;
     667                 :        496 :     assertClause(node.negate(), clause1);
     668                 :        496 :     SatClause clause2(2);
     669                 :        496 :     clause2[0] = p;
     670                 :        496 :     clause2[1] = q;
     671                 :        496 :     assertClause(node.negate(), clause2);
     672                 :            :   }
     673                 :       8687 : }
     674                 :            : 
     675                 :      57856 : void CnfStream::convertAndAssertImplies(TNode node, bool negated)
     676                 :            : {
     677 [ -  + ][ -  + ]:      57856 :   Assert(node.getKind() == Kind::IMPLIES);
                 [ -  - ]
     678         [ +  - ]:     115712 :   Trace("cnf") << "CnfStream::convertAndAssertImplies(" << node
     679         [ -  - ]:      57856 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     680         [ +  + ]:      57856 :   if (!negated) {
     681                 :            :     // p => q
     682                 :      57677 :     SatLiteral p = toCNF(node[0], false);
     683                 :      57677 :     SatLiteral q = toCNF(node[1], false);
     684                 :            :     // Construct the clause ~p || q
     685                 :      57677 :     SatClause clause(2);
     686                 :      57677 :     clause[0] = ~p;
     687                 :      57677 :     clause[1] = q;
     688                 :      57677 :     assertClause(node, clause);
     689                 :            :   } else {// Construct the
     690                 :            :     // !(p => q) is the same as (p && ~q)
     691                 :        179 :     convertAndAssert(node[0], false);
     692                 :        179 :     convertAndAssert(node[1], true);
     693                 :            :   }
     694                 :      57856 : }
     695                 :            : 
     696                 :      19781 : void CnfStream::convertAndAssertIte(TNode node, bool negated)
     697                 :            : {
     698 [ -  + ][ -  + ]:      19781 :   Assert(node.getKind() == Kind::ITE);
                 [ -  - ]
     699         [ +  - ]:      39562 :   Trace("cnf") << "CnfStream::convertAndAssertIte(" << node
     700         [ -  - ]:      19781 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     701                 :            :   // ITE(p, q, r)
     702                 :      19781 :   SatLiteral p = toCNF(node[0], false);
     703                 :      19781 :   SatLiteral q = toCNF(node[1], negated);
     704                 :      19781 :   SatLiteral r = toCNF(node[2], negated);
     705                 :            :   // Construct the clauses:
     706                 :            :   // (p => q) and (!p => r)
     707                 :            :   //
     708                 :            :   // Note that below q and r can be used directly because whether they are
     709                 :            :   // negated has been push to the literal definitions above
     710                 :      39562 :   Node nnode = node;
     711         [ +  + ]:      19781 :   if( negated ){
     712                 :        105 :     nnode = node.negate();
     713                 :            :   }
     714                 :      39562 :   SatClause clause1(2);
     715                 :      19781 :   clause1[0] = ~p;
     716                 :      19781 :   clause1[1] = q;
     717                 :      19781 :   assertClause(nnode, clause1);
     718                 :      19781 :   SatClause clause2(2);
     719                 :      19781 :   clause2[0] = p;
     720                 :      19781 :   clause2[1] = r;
     721                 :      19781 :   assertClause(nnode, clause2);
     722                 :      19781 : }
     723                 :            : 
     724                 :            : // At the top level we must ensure that all clauses that are asserted are
     725                 :            : // not unit, except for the direct assertions. This allows us to remove the
     726                 :            : // clauses later when they are not needed anymore (lemmas for example).
     727                 :     713798 : void CnfStream::convertAndAssert(TNode node, bool removable, bool negated)
     728                 :            : {
     729         [ +  - ]:    1427600 :   Trace("cnf") << "convertAndAssert(" << node
     730         [ -  - ]:          0 :                << ", negated = " << (negated ? "true" : "false")
     731         [ -  - ]:     713798 :                << ", removable = " << (removable ? "true" : "false") << ")\n";
     732                 :     713798 :   d_removable = removable;
     733                 :     713814 :   TimerStat::CodeTimer codeTimer(d_stats.d_cnfConversionTime, true);
     734                 :     713814 :   convertAndAssert(node, negated);
     735                 :     713782 : }
     736                 :            : 
     737                 :     865928 : void CnfStream::convertAndAssert(TNode node, bool negated)
     738                 :            : {
     739         [ +  - ]:    1731860 :   Trace("cnf") << "convertAndAssert(" << node
     740         [ -  - ]:     865928 :                << ", negated = " << (negated ? "true" : "false") << ")\n";
     741                 :            : 
     742                 :     865928 :   resourceManager()->spendResource(Resource::CnfStep);
     743                 :            : 
     744 [ +  + ][ +  + ]:     865928 :   switch(node.getKind()) {
         [ +  + ][ +  + ]
     745                 :     100210 :     case Kind::AND: convertAndAssertAnd(node, negated); break;
     746                 :     233266 :     case Kind::OR: convertAndAssertOr(node, negated); break;
     747                 :         70 :     case Kind::XOR: convertAndAssertXor(node, negated); break;
     748                 :      57856 :     case Kind::IMPLIES: convertAndAssertImplies(node, negated); break;
     749                 :      19781 :     case Kind::ITE: convertAndAssertIte(node, negated); break;
     750                 :     105895 :     case Kind::NOT: convertAndAssert(node[0], !negated); break;
     751                 :      85066 :     case Kind::EQUAL:
     752         [ +  + ]:      85066 :       if (node[0].getType().isBoolean())
     753                 :            :       {
     754                 :       8687 :         convertAndAssertIff(node, negated);
     755                 :       8687 :         break;
     756                 :            :       }
     757                 :            :       CVC5_FALLTHROUGH;
     758                 :            :     default:
     759                 :            :     {
     760                 :     340191 :       Node nnode = node;
     761         [ +  + ]:     340175 :       if (negated)
     762                 :            :       {
     763                 :      99559 :         nnode = node.negate();
     764                 :            :       }
     765                 :            :       // Atoms
     766                 :     340191 :       assertClause(nnode, toCNF(node, negated));
     767                 :            :   }
     768                 :     340159 :     break;
     769                 :            :   }
     770                 :     865900 : }
     771                 :            : 
     772                 :      92096 : CnfStream::Statistics::Statistics(StatisticsRegistry& sr,
     773                 :      92096 :                                   const std::string& name)
     774                 :            :     : d_cnfConversionTime(
     775                 :      92096 :         sr.registerTimer(name + "::CnfStream::cnfConversionTime")),
     776                 :      92096 :       d_numAtoms(sr.registerInt(name + "::CnfStream::numAtoms"))
     777                 :            : {
     778                 :      92096 : }
     779                 :            : 
     780                 :          0 : void CnfStream::dumpDimacs(std::ostream& out, const std::vector<Node>& clauses)
     781                 :            : {
     782                 :          0 :   std::vector<Node> auxUnits;
     783                 :          0 :   dumpDimacs(out, clauses, auxUnits);
     784                 :          0 : }
     785                 :            : 
     786                 :          0 : void CnfStream::dumpDimacs(std::ostream& out,
     787                 :            :                            const std::vector<Node>& clauses,
     788                 :            :                            const std::vector<Node>& auxUnits)
     789                 :            : {
     790                 :          0 :   std::stringstream dclauses;
     791                 :          0 :   SatVariable maxVar = 0;
     792         [ -  - ]:          0 :   for (size_t j = 0; j < 2; j++)
     793                 :            :   {
     794         [ -  - ]:          0 :     const std::vector<Node>& cls = j == 0 ? clauses : auxUnits;
     795         [ -  - ]:          0 :     for (const Node& i : cls)
     796                 :            :     {
     797                 :          0 :       std::vector<Node> lits;
     798                 :          0 :       if (j == 0 && i.getKind() == Kind::OR)
     799                 :            :       {
     800                 :            :         // print as clause if not an auxiliary unit
     801                 :          0 :         lits.insert(lits.end(), i.begin(), i.end());
     802                 :            :       }
     803                 :            :       else
     804                 :            :       {
     805                 :          0 :         lits.push_back(i);
     806                 :            :       }
     807         [ -  - ]:          0 :       Trace("dimacs-debug") << "Print " << i << std::endl;
     808         [ -  - ]:          0 :       for (const Node& l : lits)
     809                 :            :       {
     810                 :          0 :         bool negated = l.getKind() == Kind::NOT;
     811         [ -  - ]:          0 :         const Node& atom = negated ? l[0] : l;
     812                 :          0 :         SatLiteral lit = getLiteral(atom);
     813                 :          0 :         SatVariable v = lit.getSatVariable();
     814         [ -  - ]:          0 :         maxVar = v > maxVar ? v : maxVar;
     815         [ -  - ]:          0 :         dclauses << (negated ? "-" : "") << v << " ";
     816                 :            :       }
     817                 :          0 :       dclauses << "0" << std::endl;
     818                 :            :     }
     819                 :            :   }
     820                 :            : 
     821                 :          0 :   out << "p cnf " << maxVar << " " << (clauses.size() + auxUnits.size())
     822                 :          0 :       << std::endl;
     823                 :          0 :   out << dclauses.str();
     824                 :          0 : }
     825                 :            : 
     826                 :            : }  // namespace prop
     827                 :            : }  // namespace cvc5::internal

Generated by: LCOV version 1.14