GCC Code Coverage Report
Directory: . Exec Total Coverage
File: src/theory/strings/normal_form.cpp Lines: 84 84 100.0 %
Date: 2021-05-22 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
97686
void NormalForm::init(Node base)
31
{
32
97686
  Assert(base.getType().isStringLike());
33
97686
  Assert(base.getKind() != STRING_CONCAT);
34
97686
  d_base = base;
35
97686
  d_nf.clear();
36
97686
  d_isRev = false;
37
97686
  d_exp.clear();
38
97686
  d_expDep.clear();
39
40
  // add to normal form
41
97686
  if (!base.isConst() || Word::getLength(base) > 0)
42
  {
43
88335
    d_nf.push_back(base);
44
  }
45
97686
}
46
47
35932
void NormalForm::reverse()
48
{
49
35932
  std::reverse(d_nf.begin(), d_nf.end());
50
35932
  d_isRev = !d_isRev;
51
35932
}
52
53
3716
void NormalForm::splitConstant(unsigned index, Node c1, Node c2)
54
{
55
3716
  Assert(Rewriter::rewrite(NodeManager::currentNM()->mkNode(
56
             STRING_CONCAT, d_isRev ? c2 : c1, d_isRev ? c1 : c2))
57
         == d_nf[index]);
58
3716
  d_nf.insert(d_nf.begin() + index + 1, c2);
59
3716
  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
8255
  for (const std::pair<const Node, std::map<bool, unsigned> >& pe : d_expDep)
65
  {
66
13617
    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
9078
      Assert(pep.second >= 0 && pep.second <= d_nf.size());
71
9078
      bool increment = (pep.first == d_isRev)
72
13617
                           ? pep.second > index
73
13617
                           : (d_nf.size() - 1 - pep.second) < index;
74
9078
      if (increment)
75
      {
76
2947
        d_expDep[pe.first][pep.first] = pep.second + 1;
77
      }
78
    }
79
  }
80
3716
}
81
82
303454
void NormalForm::addToExplanation(Node exp,
83
                                  unsigned new_val,
84
                                  unsigned new_rev_val)
85
{
86
303454
  Assert(!exp.isConst());
87
303454
  if (std::find(d_exp.begin(), d_exp.end(), exp) == d_exp.end())
88
  {
89
302256
    d_exp.push_back(exp);
90
  }
91
910362
  for (unsigned k = 0; k < 2; k++)
92
  {
93
606908
    unsigned val = k == 0 ? new_val : new_rev_val;
94
606908
    std::map<bool, unsigned>::iterator itned = d_expDep[exp].find(k == 1);
95
606908
    if (itned == d_expDep[exp].end())
96
    {
97
1209024
      Trace("strings-process-debug")
98
604512
          << "Deps : set dependency on " << exp << " to " << val
99
604512
          << " isRev=" << (k == 0) << std::endl;
100
604512
      d_expDep[exp][k == 1] = val;
101
    }
102
    else
103
    {
104
4792
      Trace("strings-process-debug")
105
2396
          << "Deps : Multiple dependencies on " << exp << " : " << itned->second
106
2396
          << " " << 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
2396
      bool cmp = val > itned->second;
110
2396
      if (cmp == (k == 1))
111
      {
112
1198
        d_expDep[exp][k == 1] = val;
113
      }
114
    }
115
  }
116
303454
}
117
118
21776
void NormalForm::getExplanation(int index, std::vector<Node>& curr_exp)
119
{
120
41048
  if (index == -1 || !options::stringMinPrefixExplain())
121
  {
122
2504
    curr_exp.insert(curr_exp.end(), d_exp.begin(), d_exp.end());
123
2504
    return;
124
  }
125
81345
  for (const Node& exp : d_exp)
126
  {
127
62073
    int dep = static_cast<int>(d_expDep[exp][d_isRev]);
128
62073
    if (dep <= index)
129
    {
130
36551
      curr_exp.push_back(exp);
131
36551
      Trace("strings-explain-prefix-debug") << "  include : ";
132
    }
133
    else
134
    {
135
25522
      Trace("strings-explain-prefix-debug") << "  exclude : ";
136
    }
137
62073
    Trace("strings-explain-prefix-debug") << exp << std::endl;
138
  }
139
}
140
141
7264
Node NormalForm::collectConstantStringAt(size_t& index)
142
{
143
14528
  std::vector<Node> c;
144
19810
  while (d_nf.size() > index && d_nf[index].isConst())
145
  {
146
6273
    c.push_back(d_nf[index]);
147
6273
    index++;
148
  }
149
7264
  if (!c.empty())
150
  {
151
5118
    if (d_isRev)
152
    {
153
2914
      std::reverse(c.begin(), c.end());
154
    }
155
10236
    Node cc = Rewriter::rewrite(utils::mkConcat(c, c[0].getType()));
156
5118
    Assert(cc.isConst());
157
5118
    return cc;
158
  }
159
  else
160
  {
161
2146
    return Node::null();
162
  }
163
}
164
165
10888
void NormalForm::getExplanationForPrefixEq(NormalForm& nfi,
166
                                           NormalForm& nfj,
167
                                           int index_i,
168
                                           int index_j,
169
                                           std::vector<Node>& curr_exp)
170
{
171
10888
  Assert(nfi.d_isRev == nfj.d_isRev);
172
21776
  Trace("strings-explain-prefix")
173
10888
      << "Get explanation for prefix " << index_i << ", " << index_j
174
10888
      << ", reverse = " << nfi.d_isRev << std::endl;
175
  // get explanations
176
10888
  nfi.getExplanation(index_i, curr_exp);
177
10888
  nfj.getExplanation(index_j, curr_exp);
178
21776
  Trace("strings-explain-prefix")
179
21776
      << "Included " << curr_exp.size() << " / "
180
10888
      << (nfi.d_exp.size() + nfj.d_exp.size()) << std::endl;
181
  // add explanation for why they are equal
182
21776
  Node eq = nfi.d_base.eqNode(nfj.d_base);
183
10888
  curr_exp.push_back(eq);
184
10888
}
185
186
}  // namespace strings
187
}  // namespace theory
188
47466
}  // namespace cvc5