summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorAndres Noetzli <andres.noetzli@gmail.com>2020-09-16 13:36:05 -0700
committerGitHub <noreply@github.com>2020-09-16 15:36:05 -0500
commit70bfc4f67bcad32fa1b9b581b71b3a2d70e14d7e (patch)
tree3dc2c6217ad53dbf6a6fe889bb51b495a2a44e81 /src
parent2c2f05c96e021006275a2bc70b9ede70b280616d (diff)
Only rewrite replace_re(_all) if regexp is const (#5075)
Fixes #5074. This commit fixes an issue in the rewriter of str.replace_re and str.replace_re_all. For both operators, we do two kinds of rewrites: (1) If the first argument is a constant, then we check whether the regular expression appears in that constant and (2) we check whether the regular expression matches the empty string. Both of those checks were assuming a constant regular expression but there was no guard in place for it. This commit adds the missing guard.
Diffstat (limited to 'src')
-rw-r--r--src/theory/strings/sequences_rewriter.cpp86
1 files changed, 46 insertions, 40 deletions
diff --git a/src/theory/strings/sequences_rewriter.cpp b/src/theory/strings/sequences_rewriter.cpp
index f05c32356..29c0aa2d5 100644
--- a/src/theory/strings/sequences_rewriter.cpp
+++ b/src/theory/strings/sequences_rewriter.cpp
@@ -3021,31 +3021,34 @@ Node SequencesRewriter::rewriteReplaceRe(Node node)
Node y = node[1];
Node z = node[2];
- if (x.isConst())
+ if (RegExpEntail::isConstRegExp(y))
{
- // str.replace_re("ZABCZ", re.++("A", _*, "C"), y) ---> "Z" ++ y ++ "Z"
- std::pair<size_t, size_t> match = firstMatch(x, y);
- if (match.first != string::npos)
+ if (x.isConst())
{
- String s = x.getConst<String>();
- Node ret = nm->mkNode(STRING_CONCAT,
- nm->mkConst(s.substr(0, match.first)),
- z,
- nm->mkConst(s.substr(match.second)));
- return returnRewrite(node, ret, Rewrite::REPLACE_RE_EVAL);
+ // str.replace_re("ZABCZ", re.++("A", _*, "C"), y) ---> "Z" ++ y ++ "Z"
+ std::pair<size_t, size_t> match = firstMatch(x, y);
+ if (match.first != string::npos)
+ {
+ String s = x.getConst<String>();
+ Node ret = nm->mkNode(STRING_CONCAT,
+ nm->mkConst(s.substr(0, match.first)),
+ z,
+ nm->mkConst(s.substr(match.second)));
+ return returnRewrite(node, ret, Rewrite::REPLACE_RE_EVAL);
+ }
+ else
+ {
+ return returnRewrite(node, x, Rewrite::REPLACE_RE_EVAL);
+ }
}
- else
+ // str.replace_re( x, y, z ) ---> z ++ x if "" in y ---> true
+ String emptyStr("");
+ if (RegExpEntail::testConstStringInRegExp(emptyStr, 0, y))
{
- return returnRewrite(node, x, Rewrite::REPLACE_RE_EVAL);
+ Node ret = nm->mkNode(STRING_CONCAT, z, x);
+ return returnRewrite(node, ret, Rewrite::REPLACE_RE_EMP_RE);
}
}
- // str.replace_re( x, y, z ) ---> z ++ x if "" in y ---> true
- String emptyStr("");
- if (RegExpEntail::testConstStringInRegExp(emptyStr, 0, y))
- {
- Node ret = nm->mkNode(STRING_CONCAT, z, x);
- return returnRewrite(node, ret, Rewrite::REPLACE_RE_EMP_RE);
- }
return node;
}
@@ -3057,31 +3060,34 @@ Node SequencesRewriter::rewriteReplaceReAll(Node node)
Node y = node[1];
Node z = node[2];
- if (x.isConst())
+ if (RegExpEntail::isConstRegExp(y))
{
- // str.replace_re_all("ZABCZAB", re.++("A", _*, "C"), y) --->
- // "Z" ++ y ++ "Z" ++ y
- TypeNode t = x.getType();
- Node emp = Word::mkEmptyWord(t);
- Node yp = Rewriter::rewrite(
- nm->mkNode(REGEXP_DIFF, y, nm->mkNode(STRING_TO_REGEXP, emp)));
- std::vector<Node> res;
- String rem = x.getConst<String>();
- std::pair<size_t, size_t> match(0, 0);
- while (rem.size() >= 0)
+ if (x.isConst())
{
- match = firstMatch(nm->mkConst(rem), yp);
- if (match.first == string::npos)
- {
- break;
+ // str.replace_re_all("ZABCZAB", re.++("A", _*, "C"), y) --->
+ // "Z" ++ y ++ "Z" ++ y
+ TypeNode t = x.getType();
+ Node emp = Word::mkEmptyWord(t);
+ Node yp = Rewriter::rewrite(
+ nm->mkNode(REGEXP_DIFF, y, nm->mkNode(STRING_TO_REGEXP, emp)));
+ std::vector<Node> res;
+ String rem = x.getConst<String>();
+ std::pair<size_t, size_t> match(0, 0);
+ while (rem.size() >= 0)
+ {
+ match = firstMatch(nm->mkConst(rem), yp);
+ if (match.first == string::npos)
+ {
+ break;
+ }
+ res.push_back(nm->mkConst(rem.substr(0, match.first)));
+ res.push_back(z);
+ rem = rem.substr(match.second);
}
- res.push_back(nm->mkConst(rem.substr(0, match.first)));
- res.push_back(z);
- rem = rem.substr(match.second);
+ res.push_back(nm->mkConst(rem));
+ Node ret = utils::mkConcat(res, t);
+ return returnRewrite(node, ret, Rewrite::REPLACE_RE_ALL_EVAL);
}
- res.push_back(nm->mkConst(rem));
- Node ret = utils::mkConcat(res, t);
- return returnRewrite(node, ret, Rewrite::REPLACE_RE_ALL_EVAL);
}
return node;
generated by cgit on debian on lair
contact matthew@masot.net with questions or feedback