GCC Code Coverage Report
Directory: . Exec Total Coverage
File: src/theory/strings/normal_form.cpp Lines: 84 84 100.0 %
Date: 2021-08-01 Branches: 150 342 43.9 %

Line Exec Source
1
/******************************************************************************
2
 * Top contributors (to current version):
3
 *   Andrew Reynolds, Andres Noetzli, Gereon Kremer
4
 *
5
 * This file is part of the cvc5 project.
6
 *
7
 * Copyright (c) 2009-2021 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
 * Implementation of normal form data structure for the theory of strings.
14
 */
15
16
#include "theory/strings/normal_form.h"
17
18
#include "options/strings_options.h"
19
#include "theory/rewriter.h"
20
#include "theory/strings/theory_strings_utils.h"
21
#include "theory/strings/word.h"
22
23
using namespace std;
24
using namespace cvc5::kind;
25
26
namespace cvc5 {
27
namespace theory {
28
namespace strings {
29
30
113951
void NormalForm::init(Node base)
31
{
32
113951
  Assert(base.getType().isStringLike());
33
113951
  Assert(base.getKind() != STRING_CONCAT);
34
113951
  d_base = base;
35
113951
  d_nf.clear();
36
113951
  d_isRev = false;
37
113951
  d_exp.clear();
38
113951
  d_expDep.clear();
39
40
  // add to normal form
41
113951
  if (!base.isConst() || Word::getLength(base) > 0)
42
  {
43
102052
    d_nf.push_back(base);
44
  }
45
113951
}
46
47
42788
void NormalForm::reverse()
48
{
49
42788
  std::reverse(d_nf.begin(), d_nf.end());
50
42788
  d_isRev = !d_isRev;
51
42788
}
52
53
5195
void NormalForm::splitConstant(unsigned index, Node c1, Node c2)
54
{
55
5195
  Assert(Rewriter::rewrite(NodeManager::currentNM()->mkNode(
56
             STRING_CONCAT, d_isRev ? c2 : c1, d_isRev ? c1 : c2))
57
         == d_nf[index]);
58
5195
  d_nf.insert(d_nf.begin() + index + 1, c2);
59
5195
  d_nf[index] = c1;
60
  // update the dependency indices
61
  // notice this is not critical for soundness: not doing the below incrementing
62
  // will only lead to overapproximating when antecedants are required in
63
  // explanations
64
16391
  for (const std::pair<const Node, std::map<bool, unsigned> >& pe : d_expDep)
65
  {
66
33588
    for (const auto& pep : pe.second)
67
    {
68
      // See if this can be incremented: it can if this literal is not relevant
69
      // to the current index, and hence it is not relevant for both c1 and c2.
70
22392
      Assert(pep.second >= 0 && pep.second <= d_nf.size());
71
22392
      bool increment = (pep.first == d_isRev)
72
33588
                           ? pep.second > index
73
33588
                           : (d_nf.size() - 1 - pep.second) < index;
74
22392
      if (increment)
75
      {
76
9103
        d_expDep[pe.first][pep.first] = pep.second + 1;
77
      }
78
    }
79
  }
80
5195
}
81
82
353512
void NormalForm::addToExplanation(Node exp,
83
                                  unsigned new_val,
84
                                  unsigned new_rev_val)
85
{
86
353512
  Assert(!exp.isConst());
87
353512
  if (std::find(d_exp.begin(), d_exp.end(), exp) == d_exp.end())
88
  {
89
352127
    d_exp.push_back(exp);
90
  }
91
1060536
  for (unsigned k = 0; k < 2; k++)
92
  {
93
707024
    unsigned val = k == 0 ? new_val : new_rev_val;
94
707024
    std::map<bool, unsigned>::iterator itned = d_expDep[exp].find(k == 1);
95
707024
    if (itned == d_expDep[exp].end())
96
    {
97
1408508
      Trace("strings-process-debug")
98
704254
          << "Deps : set dependency on " << exp << " to " << val
99
704254
          << " isRev=" << (k == 0) << std::endl;
100
704254
      d_expDep[exp][k == 1] = val;
101
    }
102
    else
103
    {
104
5540
      Trace("strings-process-debug")
105
2770
          << "Deps : Multiple dependencies on " << exp << " : " << itned->second
106
2770
          << " " << val << " isRev=" << (k == 0) << std::endl;
107
      // if we already have a dependency (in the case of non-linear string
108
      // equalities), it is min/max
109
2770
      bool cmp = val > itned->second;
110
2770
      if (cmp == (k == 1))
111
      {
112
1385
        d_expDep[exp][k == 1] = val;
113
      }
114
    }
115
  }
116
353512
}
117
118
26292
void NormalForm::getExplanation(int index, std::vector<Node>& curr_exp)
119
{
120
49132
  if (index == -1 || !options::stringMinPrefixExplain())
121
  {
122
3452
    curr_exp.insert(curr_exp.end(), d_exp.begin(), d_exp.end());
123
3452
    return;
124
  }
125
94894
  for (const Node& exp : d_exp)
126
  {
127
72054
    int dep = static_cast<int>(d_expDep[exp][d_isRev]);
128
72054
    if (dep <= index)
129
    {
130
41681
      curr_exp.push_back(exp);
131
41681
      Trace("strings-explain-prefix-debug") << "  include : ";
132
    }
133
    else
134
    {
135
30373
      Trace("strings-explain-prefix-debug") << "  exclude : ";
136
    }
137
72054
    Trace("strings-explain-prefix-debug") << exp << std::endl;
138
  }
139
}
140
141
8671
Node NormalForm::collectConstantStringAt(size_t& index)
142
{
143
17342
  std::vector<Node> c;
144
23411
  while (d_nf.size() > index && d_nf[index].isConst())
145
  {
146
7370
    c.push_back(d_nf[index]);
147
7370
    index++;
148
  }
149
8671
  if (!c.empty())
150
  {
151
5994
    if (d_isRev)
152
    {
153
3388
      std::reverse(c.begin(), c.end());
154
    }
155
11988
    Node cc = Rewriter::rewrite(utils::mkConcat(c, c[0].getType()));
156
5994
    Assert(cc.isConst());
157
5994
    return cc;
158
  }
159
  else
160
  {
161
2677
    return Node::null();
162
  }
163
}
164
165
13146
void NormalForm::getExplanationForPrefixEq(NormalForm& nfi,
166
                                           NormalForm& nfj,
167
                                           int index_i,
168
                                           int index_j,
169
                                           std::vector<Node>& curr_exp)
170
{
171
13146
  Assert(nfi.d_isRev == nfj.d_isRev);
172
26292
  Trace("strings-explain-prefix")
173
13146
      << "Get explanation for prefix " << index_i << ", " << index_j
174
13146
      << ", reverse = " << nfi.d_isRev << std::endl;
175
  // get explanations
176
13146
  nfi.getExplanation(index_i, curr_exp);
177
13146
  nfj.getExplanation(index_j, curr_exp);
178
26292
  Trace("strings-explain-prefix")
179
26292
      << "Included " << curr_exp.size() << " / "
180
13146
      << (nfi.d_exp.size() + nfj.d_exp.size()) << std::endl;
181
  // add explanation for why they are equal
182
26292
  Node eq = nfi.d_base.eqNode(nfj.d_base);
183
13146
  curr_exp.push_back(eq);
184
13146
}
185
186
}  // namespace strings
187
}  // namespace theory
188
52120
}  // namespace cvc5