mirror of
https://github.com/Z3Prover/z3
synced 2025-04-06 17:44:08 +00:00
address divergence in the case of shared theory symbols. Codeplex issue 147, thanks to George Karpenkov
Signed-off-by: Nikolaj Bjorner <nbjorner@microsoft.com>
This commit is contained in:
commit
08cb8b8de8
3
.gitignore
vendored
3
.gitignore
vendored
|
@ -65,3 +65,6 @@ src/util/version.h
|
||||||
src/api/java/Native.cpp
|
src/api/java/Native.cpp
|
||||||
src/api/java/Native.java
|
src/api/java/Native.java
|
||||||
src/api/java/enumerations/*.java
|
src/api/java/enumerations/*.java
|
||||||
|
*.bak
|
||||||
|
doc/api
|
||||||
|
doc/code
|
||||||
|
|
|
@ -1,8 +1,22 @@
|
||||||
RELEASE NOTES
|
RELEASE NOTES
|
||||||
|
|
||||||
|
Version 4.3.3
|
||||||
|
=============
|
||||||
|
|
||||||
|
- Fixed bug in floating point models
|
||||||
|
|
||||||
Version 4.3.2
|
Version 4.3.2
|
||||||
=============
|
=============
|
||||||
|
|
||||||
|
- Added preliminary support for the theory of floating point numbers (tactics qffpa, qffpabv, and logics QF_FPA, QF_FPABV).
|
||||||
|
|
||||||
|
- Added the interpolation features of iZ3, which are now integrated into the Z3.
|
||||||
|
|
||||||
|
- Fixed a multitude of bugs and inconsistencies that were reported to us either in person, by email, or on Codeplex. Of those that we do have records of, we would like to express our gratitude to:
|
||||||
|
Vladimir Klebanov, Konrad Jamrozik, Nuno Lopes, Carsten Ruetz, Esteban Pavese, Tomer Weiss, Ilya Mironov, Gabriele Paganelli, Levent Erkok, Fabian Emmes, David Cok, Etienne Kneuss, Arlen Cox, Matt Lewis, Carsten Otto, Paul Jackson, David Monniaux, Markus Rabe, Martin Pluecker, Jasmin Blanchette, Jules Villard, Andrew Gacek, George Karpenkov, Joerg Pfaehler, and Pablo Aledo
|
||||||
|
as well as the following Codeplex users that either reported bugs or took part in discussions:
|
||||||
|
xor88, parno, gario, Bauna, GManNickG, hanwentao, dinu09, fhowar, Cici, chinissai, barak_cohen, tvalentyn, krikunts, sukyoung, daramos, snedunuri, rajtendulkar, sonertari, nick8325, dvitek, amdragon, Beatgodes, dmonniaux, nickolai, DameNingen, mangpo, ttsiodras, blurium, sbrickey, pcodemod, indranilsaha, apanda, hougaardj, yoff, EfForEffort, Ansotegui, scottgw, viorelpreoteasa, idudka, c2855337, gario, jnfoster, omarmrivas, switicus, vosandi, foens, yzwwf, Heizmann, znajem, ilyagri, hougaardj, cliguda, rgrig, 92c849c1ccc707173, edmcman, cipher1024, MichaelvW, hellok, n00b42, ic3guy, Adorf, tvcsantos, zilongwang, Elarnon, immspw, jbridge99, danliew, zverlov, petross, jmh93, dradorf, fniksic, Heyji, cxcfan, henningg, wxlfrank, rvprasad, MovGP0, jackie1015, cowang, ffaghih, sanpra1989, gzchenyin, baitman, xjtulixiangyang, andreis, trucnguyenlam, erizzi, hanhchi, qsp, windypan, vadave, gradanne, SamWot, gsingh93, manjeetdahiya, zverlov, RaLa, and regehr.
|
||||||
|
|
||||||
- New parameter setting infrastructure. Now, it is possible to set parameter for Z3 internal modules. Several parameter names changed. Execute `z3 -p` for the new parameter list.
|
- New parameter setting infrastructure. Now, it is possible to set parameter for Z3 internal modules. Several parameter names changed. Execute `z3 -p` for the new parameter list.
|
||||||
|
|
||||||
- Added get_version() and get_version_string() to Z3Py
|
- Added get_version() and get_version_string() to Z3Py
|
||||||
|
|
|
@ -27,6 +27,10 @@ try:
|
||||||
shutil.copyfile('website.dox', 'tmp/website.dox')
|
shutil.copyfile('website.dox', 'tmp/website.dox')
|
||||||
shutil.copyfile('../src/api/python/z3.py', 'tmp/z3py.py')
|
shutil.copyfile('../src/api/python/z3.py', 'tmp/z3py.py')
|
||||||
cleanup_API('../src/api/z3_api.h', 'tmp/z3_api.h')
|
cleanup_API('../src/api/z3_api.h', 'tmp/z3_api.h')
|
||||||
|
cleanup_API('../src/api/z3_algebraic.h', 'tmp/z3_algebraic.h')
|
||||||
|
cleanup_API('../src/api/z3_polynomial.h', 'tmp/z3_polynomial.h')
|
||||||
|
cleanup_API('../src/api/z3_rcf.h', 'tmp/z3_rcf.h')
|
||||||
|
cleanup_API('../src/api/z3_interp.h', 'tmp/z3_interp.h')
|
||||||
|
|
||||||
print "Removed annotations from z3_api.h."
|
print "Removed annotations from z3_api.h."
|
||||||
try:
|
try:
|
||||||
|
@ -38,6 +42,10 @@ try:
|
||||||
exit(1)
|
exit(1)
|
||||||
print "Generated C and .NET API documentation."
|
print "Generated C and .NET API documentation."
|
||||||
os.remove('tmp/z3_api.h')
|
os.remove('tmp/z3_api.h')
|
||||||
|
os.remove('tmp/z3_algebraic.h')
|
||||||
|
os.remove('tmp/z3_polynomial.h')
|
||||||
|
os.remove('tmp/z3_rcf.h')
|
||||||
|
os.remove('tmp/z3_interp.h')
|
||||||
print "Removed temporary file z3_api.h."
|
print "Removed temporary file z3_api.h."
|
||||||
os.remove('tmp/website.dox')
|
os.remove('tmp/website.dox')
|
||||||
print "Removed temporary file website.dox"
|
print "Removed temporary file website.dox"
|
||||||
|
|
|
@ -15,5 +15,5 @@
|
||||||
- <a class="el" href="class_microsoft_1_1_z3_1_1_context.html">.NET API</a>
|
- <a class="el" href="class_microsoft_1_1_z3_1_1_context.html">.NET API</a>
|
||||||
- <a class="el" href="namespacecom_1_1microsoft_1_1z3.html">Java API</a>
|
- <a class="el" href="namespacecom_1_1microsoft_1_1z3.html">Java API</a>
|
||||||
- <a class="el" href="namespacez3py.html">Python API</a> (also available in <a class="el" href="z3.html">pydoc format</a>).
|
- <a class="el" href="namespacez3py.html">Python API</a> (also available in <a class="el" href="z3.html">pydoc format</a>).
|
||||||
- Try Z3 online at <a href="http://rise4fun.com/z3py">RiSE4Fun</a> using <a href="http://rise4fun.com/z3py">Python</a> or <a href="http://rise4fun.com/z3">SMT 2.0</a>.
|
- Try Z3 online at <a href="http://rise4fun.com/z3">RiSE4Fun</a>.
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -363,7 +363,7 @@ TYPEDEF_HIDES_STRUCT = NO
|
||||||
# 2^(16+SYMBOL_CACHE_SIZE). The valid range is 0..9, the default is 0,
|
# 2^(16+SYMBOL_CACHE_SIZE). The valid range is 0..9, the default is 0,
|
||||||
# corresponding to a cache size of 2^16 = 65536 symbols.
|
# corresponding to a cache size of 2^16 = 65536 symbols.
|
||||||
|
|
||||||
SYMBOL_CACHE_SIZE = 0
|
# SYMBOL_CACHE_SIZE = 0
|
||||||
|
|
||||||
# Similar to the SYMBOL_CACHE_SIZE the size of the symbol lookup cache can be
|
# Similar to the SYMBOL_CACHE_SIZE the size of the symbol lookup cache can be
|
||||||
# set using LOOKUP_CACHE_SIZE. This cache is used to resolve symbols given
|
# set using LOOKUP_CACHE_SIZE. This cache is used to resolve symbols given
|
||||||
|
@ -708,7 +708,11 @@ INPUT_ENCODING = UTF-8
|
||||||
# *.f90 *.f *.for *.vhd *.vhdl
|
# *.f90 *.f *.for *.vhd *.vhdl
|
||||||
|
|
||||||
FILE_PATTERNS = website.dox \
|
FILE_PATTERNS = website.dox \
|
||||||
z3_api.h \
|
z3_api.h \
|
||||||
|
z3_algebraic.h \
|
||||||
|
z3_polynomial.h \
|
||||||
|
z3_rcf.h \
|
||||||
|
z3_interp.h \
|
||||||
z3++.h \
|
z3++.h \
|
||||||
z3py.py \
|
z3py.py \
|
||||||
ApplyResult.cs \
|
ApplyResult.cs \
|
||||||
|
@ -1477,13 +1481,13 @@ XML_OUTPUT = xml
|
||||||
# which can be used by a validating XML parser to check the
|
# which can be used by a validating XML parser to check the
|
||||||
# syntax of the XML files.
|
# syntax of the XML files.
|
||||||
|
|
||||||
XML_SCHEMA =
|
# XML_SCHEMA =
|
||||||
|
|
||||||
# The XML_DTD tag can be used to specify an XML DTD,
|
# The XML_DTD tag can be used to specify an XML DTD,
|
||||||
# which can be used by a validating XML parser to check the
|
# which can be used by a validating XML parser to check the
|
||||||
# syntax of the XML files.
|
# syntax of the XML files.
|
||||||
|
|
||||||
XML_DTD =
|
# XML_DTD =
|
||||||
|
|
||||||
# If the XML_PROGRAMLISTING tag is set to YES Doxygen will
|
# If the XML_PROGRAMLISTING tag is set to YES Doxygen will
|
||||||
# dump the program listings (including syntax highlighting
|
# dump the program listings (including syntax highlighting
|
||||||
|
@ -1699,7 +1703,7 @@ DOT_NUM_THREADS = 0
|
||||||
# the DOTFONTPATH environment variable or by setting DOT_FONTPATH to the
|
# the DOTFONTPATH environment variable or by setting DOT_FONTPATH to the
|
||||||
# directory containing the font.
|
# directory containing the font.
|
||||||
|
|
||||||
DOT_FONTNAME = FreeSans
|
# DOT_FONTNAME = FreeSans
|
||||||
|
|
||||||
# The DOT_FONTSIZE tag can be used to set the size of the font of dot graphs.
|
# The DOT_FONTSIZE tag can be used to set the size of the font of dot graphs.
|
||||||
# The default size is 10pt.
|
# The default size is 10pt.
|
||||||
|
|
|
@ -21,6 +21,7 @@ void demorgan() {
|
||||||
// adding the negation of the conjecture as a constraint.
|
// adding the negation of the conjecture as a constraint.
|
||||||
s.add(!conjecture);
|
s.add(!conjecture);
|
||||||
std::cout << s << "\n";
|
std::cout << s << "\n";
|
||||||
|
std::cout << s.to_smt2() << "\n";
|
||||||
switch (s.check()) {
|
switch (s.check()) {
|
||||||
case unsat: std::cout << "de-Morgan is valid\n"; break;
|
case unsat: std::cout << "de-Morgan is valid\n"; break;
|
||||||
case sat: std::cout << "de-Morgan is not valid\n"; break;
|
case sat: std::cout << "de-Morgan is not valid\n"; break;
|
||||||
|
|
|
@ -1,7 +1,6 @@
|
||||||
### This is work-in-progress and does not work yet.
|
A small example using the Z3 Java bindings.
|
||||||
|
|
||||||
Small example using the Z3 Java bindings.
|
To build the example, configure Z3 with the --java option to scripts/mk_make.py, build via
|
||||||
To build the example execute
|
|
||||||
make examples
|
make examples
|
||||||
in the build directory.
|
in the build directory.
|
||||||
|
|
||||||
|
@ -11,5 +10,5 @@ which can be run on Windows via
|
||||||
|
|
||||||
On Linux and FreeBSD, we must use
|
On Linux and FreeBSD, we must use
|
||||||
LD_LIBRARY_PATH=. java -cp com.microsoft.z3.jar:. JavaExample
|
LD_LIBRARY_PATH=. java -cp com.microsoft.z3.jar:. JavaExample
|
||||||
|
On OSX, the corresponding option is DYLD_LIBRARY_PATH:
|
||||||
|
DYLD_LIBRARY_PATH=. java -cp com.microsoft.z3.jar:. JavaExample
|
||||||
|
|
|
@ -9,7 +9,7 @@ from mk_util import *
|
||||||
|
|
||||||
# Z3 Project definition
|
# Z3 Project definition
|
||||||
def init_project_def():
|
def init_project_def():
|
||||||
set_version(4, 3, 2, 0)
|
set_version(4, 3, 3, 0)
|
||||||
add_lib('util', [])
|
add_lib('util', [])
|
||||||
add_lib('polynomial', ['util'], 'math/polynomial')
|
add_lib('polynomial', ['util'], 'math/polynomial')
|
||||||
add_lib('sat', ['util'])
|
add_lib('sat', ['util'])
|
||||||
|
|
|
@ -297,11 +297,13 @@ def param2javaw(p):
|
||||||
k = param_kind(p)
|
k = param_kind(p)
|
||||||
if k == OUT:
|
if k == OUT:
|
||||||
return "jobject"
|
return "jobject"
|
||||||
if k == IN_ARRAY or k == INOUT_ARRAY or k == OUT_ARRAY:
|
elif k == IN_ARRAY or k == INOUT_ARRAY or k == OUT_ARRAY:
|
||||||
if param_type(p) == INT or param_type(p) == UINT:
|
if param_type(p) == INT or param_type(p) == UINT:
|
||||||
return "jintArray"
|
return "jintArray"
|
||||||
else:
|
else:
|
||||||
return "jlongArray"
|
return "jlongArray"
|
||||||
|
elif k == OUT_MANAGED_ARRAY:
|
||||||
|
return "jlong";
|
||||||
else:
|
else:
|
||||||
return type2javaw(param_type(p))
|
return type2javaw(param_type(p))
|
||||||
|
|
||||||
|
|
|
@ -273,6 +273,9 @@ extern "C" {
|
||||||
0 // ignore params for now
|
0 // ignore params for now
|
||||||
);
|
);
|
||||||
|
|
||||||
|
for (unsigned i = 0; i < cnsts.size(); i++)
|
||||||
|
_m.dec_ref(cnsts[i]);
|
||||||
|
|
||||||
Z3_lbool status = of_lbool(_status);
|
Z3_lbool status = of_lbool(_status);
|
||||||
|
|
||||||
Z3_ast_vector_ref *v = 0;
|
Z3_ast_vector_ref *v = 0;
|
||||||
|
@ -617,7 +620,7 @@ extern "C" {
|
||||||
}
|
}
|
||||||
|
|
||||||
for (unsigned j = 0; j < num - 1; j++)
|
for (unsigned j = 0; j < num - 1; j++)
|
||||||
if (read_parents[j] == SHRT_MIN){
|
if (read_parents[j] == SHRT_MAX){
|
||||||
read_error << "formula " << j + 1 << ": unreferenced";
|
read_error << "formula " << j + 1 << ": unreferenced";
|
||||||
goto fail;
|
goto fail;
|
||||||
}
|
}
|
||||||
|
@ -717,4 +720,4 @@ Z3_lbool Z3_API Z3_interpolate(__in Z3_context ctx,
|
||||||
__in unsigned incremental,
|
__in unsigned incremental,
|
||||||
__in unsigned num_theory,
|
__in unsigned num_theory,
|
||||||
__in_ecount(num_theory) Z3_ast *theory);
|
__in_ecount(num_theory) Z3_ast *theory);
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -1314,6 +1314,26 @@ namespace z3 {
|
||||||
expr_vector assertions() const { Z3_ast_vector r = Z3_solver_get_assertions(ctx(), m_solver); check_error(); return expr_vector(ctx(), r); }
|
expr_vector assertions() const { Z3_ast_vector r = Z3_solver_get_assertions(ctx(), m_solver); check_error(); return expr_vector(ctx(), r); }
|
||||||
expr proof() const { Z3_ast r = Z3_solver_get_proof(ctx(), m_solver); check_error(); return expr(ctx(), r); }
|
expr proof() const { Z3_ast r = Z3_solver_get_proof(ctx(), m_solver); check_error(); return expr(ctx(), r); }
|
||||||
friend std::ostream & operator<<(std::ostream & out, solver const & s) { out << Z3_solver_to_string(s.ctx(), s); return out; }
|
friend std::ostream & operator<<(std::ostream & out, solver const & s) { out << Z3_solver_to_string(s.ctx(), s); return out; }
|
||||||
|
|
||||||
|
std::string to_smt2(char const* status = "unknown") {
|
||||||
|
array<Z3_ast> es(assertions());
|
||||||
|
Z3_ast const* fmls = es.ptr();
|
||||||
|
Z3_ast fml = 0;
|
||||||
|
unsigned sz = es.size();
|
||||||
|
if (sz > 0) {
|
||||||
|
--sz;
|
||||||
|
fml = fmls[sz];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
fml = ctx().bool_val(true);
|
||||||
|
}
|
||||||
|
return std::string(Z3_benchmark_to_smtlib_string(
|
||||||
|
ctx(),
|
||||||
|
"", "", status, "",
|
||||||
|
sz,
|
||||||
|
fmls,
|
||||||
|
fml));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
class goal : public object {
|
class goal : public object {
|
||||||
|
|
|
@ -19,12 +19,12 @@
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
<DebugType>full</DebugType>
|
<DebugType>full</DebugType>
|
||||||
<Optimize>false</Optimize>
|
<Optimize>false</Optimize>
|
||||||
<OutputPath>..\..\..\..\..\cwinter\bugs\z3bugs\Debug\</OutputPath>
|
<OutputPath>..\Debug\</OutputPath>
|
||||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||||
<ErrorReport>prompt</ErrorReport>
|
<ErrorReport>prompt</ErrorReport>
|
||||||
<WarningLevel>4</WarningLevel>
|
<WarningLevel>4</WarningLevel>
|
||||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
||||||
<DocumentationFile>C:\cwinter\bugs\z3bugs\Debug\Microsoft.Z3.XML</DocumentationFile>
|
<DocumentationFile>..\Debug\Microsoft.Z3.XML</DocumentationFile>
|
||||||
<CodeContractsEnableRuntimeChecking>False</CodeContractsEnableRuntimeChecking>
|
<CodeContractsEnableRuntimeChecking>False</CodeContractsEnableRuntimeChecking>
|
||||||
<CodeContractsRuntimeOnlyPublicSurface>False</CodeContractsRuntimeOnlyPublicSurface>
|
<CodeContractsRuntimeOnlyPublicSurface>False</CodeContractsRuntimeOnlyPublicSurface>
|
||||||
<CodeContractsRuntimeThrowOnFailure>True</CodeContractsRuntimeThrowOnFailure>
|
<CodeContractsRuntimeThrowOnFailure>True</CodeContractsRuntimeThrowOnFailure>
|
||||||
|
@ -254,7 +254,7 @@
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Debug|x86'">
|
||||||
<DebugSymbols>true</DebugSymbols>
|
<DebugSymbols>true</DebugSymbols>
|
||||||
<OutputPath>..\..\..\..\..\cwinter\bugs\z3bugs\Debug\</OutputPath>
|
<OutputPath>..\x86\Debug\</OutputPath>
|
||||||
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
<DefineConstants>DEBUG;TRACE</DefineConstants>
|
||||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
||||||
<DebugType>full</DebugType>
|
<DebugType>full</DebugType>
|
||||||
|
@ -266,7 +266,7 @@
|
||||||
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
<CodeAnalysisRuleSet>MinimumRecommendedRules.ruleset</CodeAnalysisRuleSet>
|
||||||
<CodeAnalysisRuleSetDirectories>;C:\Program Files (x86)\Microsoft Visual Studio 10.0\Team Tools\Static Analysis Tools\\Rule Sets</CodeAnalysisRuleSetDirectories>
|
<CodeAnalysisRuleSetDirectories>;C:\Program Files (x86)\Microsoft Visual Studio 10.0\Team Tools\Static Analysis Tools\\Rule Sets</CodeAnalysisRuleSetDirectories>
|
||||||
<CodeAnalysisRuleDirectories>;C:\Program Files (x86)\Microsoft Visual Studio 10.0\Team Tools\Static Analysis Tools\FxCop\\Rules</CodeAnalysisRuleDirectories>
|
<CodeAnalysisRuleDirectories>;C:\Program Files (x86)\Microsoft Visual Studio 10.0\Team Tools\Static Analysis Tools\FxCop\\Rules</CodeAnalysisRuleDirectories>
|
||||||
<DocumentationFile>C:\cwinter\bugs\z3bugs\Debug\Microsoft.Z3.XML</DocumentationFile>
|
<DocumentationFile>..\x86\Debug\Microsoft.Z3.XML</DocumentationFile>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'">
|
<PropertyGroup Condition="'$(Configuration)|$(Platform)' == 'Release|x86'">
|
||||||
<OutputPath>bin\x86\Release\</OutputPath>
|
<OutputPath>bin\x86\Release\</OutputPath>
|
||||||
|
|
|
@ -12,7 +12,7 @@ using System.Security.Permissions;
|
||||||
[assembly: AssemblyConfiguration("")]
|
[assembly: AssemblyConfiguration("")]
|
||||||
[assembly: AssemblyCompany("Microsoft Corporation")]
|
[assembly: AssemblyCompany("Microsoft Corporation")]
|
||||||
[assembly: AssemblyProduct("Z3")]
|
[assembly: AssemblyProduct("Z3")]
|
||||||
[assembly: AssemblyCopyright("Copyright © Microsoft Corporation 2006")]
|
[assembly: AssemblyCopyright("Copyright (C) 2006-2014 Microsoft Corporation")]
|
||||||
[assembly: AssemblyTrademark("")]
|
[assembly: AssemblyTrademark("")]
|
||||||
[assembly: AssemblyCulture("")]
|
[assembly: AssemblyCulture("")]
|
||||||
|
|
||||||
|
|
|
@ -24,28 +24,12 @@ import com.microsoft.z3.enumerations.Z3_ast_kind;
|
||||||
**/
|
**/
|
||||||
public class AST extends Z3Object
|
public class AST extends Z3Object
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Comparison operator. <param name="a">An AST</param> <param name="b">An
|
|
||||||
* AST</param>
|
|
||||||
*
|
|
||||||
* @return True if <paramref name="a"/> and <paramref name="b"/> are from
|
|
||||||
* the same context and represent the same sort; false otherwise.
|
|
||||||
**/
|
|
||||||
/* Overloaded operators are not translated. */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Comparison operator. <param name="a">An AST</param> <param name="b">An
|
|
||||||
* AST</param>
|
|
||||||
*
|
|
||||||
* @return True if <paramref name="a"/> and <paramref name="b"/> are not
|
|
||||||
* from the same context or represent different sorts; false
|
|
||||||
* otherwise.
|
|
||||||
**/
|
|
||||||
/* Overloaded operators are not translated. */
|
/* Overloaded operators are not translated. */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Object comparison.
|
* Object comparison.
|
||||||
**/
|
* <param name="o">another AST</param>
|
||||||
|
**/
|
||||||
public boolean equals(Object o)
|
public boolean equals(Object o)
|
||||||
{
|
{
|
||||||
AST casted = null;
|
AST casted = null;
|
||||||
|
|
|
@ -40,7 +40,7 @@ public class Expr extends AST
|
||||||
/**
|
/**
|
||||||
* Returns a simplified version of the expression
|
* Returns a simplified version of the expression
|
||||||
* A set of
|
* A set of
|
||||||
* parameters <param name="p" /> to configure the simplifier
|
* parameters <param name="p">a Params object</param> to configure the simplifier
|
||||||
* <seealso cref="Context.SimplifyHelp"/>
|
* <seealso cref="Context.SimplifyHelp"/>
|
||||||
**/
|
**/
|
||||||
public Expr simplify(Params p) throws Z3Exception
|
public Expr simplify(Params p) throws Z3Exception
|
||||||
|
|
|
@ -50,8 +50,7 @@ public final class Global
|
||||||
/**
|
/**
|
||||||
* Get a global (or module) parameter.
|
* Get a global (or module) parameter.
|
||||||
* <remarks>
|
* <remarks>
|
||||||
* Returns null if the parameter <param name="id"/> does not exist.
|
* Returns null if the parameter <param name="id">parameter id</param> does not exist.
|
||||||
* The caller must invoke #Z3_global_param_del_value to delete the value returned at \c param_value.
|
|
||||||
* This function cannot be invoked simultaneously from different threads without synchronization.
|
* This function cannot be invoked simultaneously from different threads without synchronization.
|
||||||
* The result string stored in param_value is stored in a shared location.
|
* The result string stored in param_value is stored in a shared location.
|
||||||
* </remarks>
|
* </remarks>
|
||||||
|
@ -70,7 +69,7 @@ public final class Global
|
||||||
* <remarks>
|
* <remarks>
|
||||||
* This command will not affect already created objects (such as tactics and solvers)
|
* This command will not affect already created objects (such as tactics and solvers)
|
||||||
* </remarks>
|
* </remarks>
|
||||||
* @seealso SetParameter
|
* <seealso cref="SetParameter"/>
|
||||||
**/
|
**/
|
||||||
public static void resetParameters()
|
public static void resetParameters()
|
||||||
{
|
{
|
||||||
|
|
|
@ -1,3 +1,6 @@
|
||||||
Java bindings
|
Java bindings
|
||||||
-------------
|
-------------
|
||||||
|
|
||||||
|
The Java bindings will be included in the Z3 build if it is configured with
|
||||||
|
the option --java to python scripts/mk_make.py. This will produce the
|
||||||
|
com.microsoft.z3.jar package in the build directory.
|
||||||
|
|
|
@ -25,27 +25,10 @@ import com.microsoft.z3.enumerations.Z3_sort_kind;
|
||||||
**/
|
**/
|
||||||
public class Sort extends AST
|
public class Sort extends AST
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Comparison operator. <param name="a">A Sort</param> <param name="b">A
|
|
||||||
* Sort</param>
|
|
||||||
*
|
|
||||||
* @return True if <paramref name="a"/> and <paramref name="b"/> are from
|
|
||||||
* the same context and represent the same sort; false otherwise.
|
|
||||||
**/
|
|
||||||
/* Overloaded operators are not translated. */
|
/* Overloaded operators are not translated. */
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Comparison operator. <param name="a">A Sort</param> <param name="b">A
|
* Equality operator for objects of type Sort. <param name="o"/>
|
||||||
* Sort</param>
|
|
||||||
*
|
|
||||||
* @return True if <paramref name="a"/> and <paramref name="b"/> are not
|
|
||||||
* from the same context or represent different sorts; false
|
|
||||||
* otherwise.
|
|
||||||
**/
|
|
||||||
/* Overloaded operators are not translated. */
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Equality operator for objects of type Sort. <param name="o"></param>
|
|
||||||
*
|
*
|
||||||
* @return
|
* @return
|
||||||
**/
|
**/
|
||||||
|
|
|
@ -1,2 +1,2 @@
|
||||||
Manifest-Version: 1.0
|
Manifest-Version: 1.0
|
||||||
Created-By: 4.3.0 (Microsoft Research LTD.)
|
Created-By: 4.3.2 (Microsoft Research LTD.)
|
||||||
|
|
|
@ -1,463 +0,0 @@
|
||||||
######################################################
|
|
||||||
# Copyright (c) 2012 Microsoft Corporation
|
|
||||||
#
|
|
||||||
# Auxiliary scripts for generating Java bindings
|
|
||||||
# from the managed API.
|
|
||||||
#
|
|
||||||
# Author: Christoph M. Wintersteiger (cwinter)
|
|
||||||
######################################################
|
|
||||||
|
|
||||||
|
|
||||||
###
|
|
||||||
# DO NOT USE THIS SCRIPT!
|
|
||||||
# This script creates a rough draft of a Java API from
|
|
||||||
# the managed API, but does not automated the process.
|
|
||||||
###
|
|
||||||
|
|
||||||
CS="../dotnet/"
|
|
||||||
EXT=".cs"
|
|
||||||
EXCLUDE=["Enumerations.cs", "Native.cs", "AssemblyInfo.cs"]
|
|
||||||
OUTDIR="com/Microsoft/Z3/"
|
|
||||||
ENUMS_FILE = "Enumerations.cs"
|
|
||||||
|
|
||||||
import os
|
|
||||||
import fileinput
|
|
||||||
import string
|
|
||||||
import re
|
|
||||||
|
|
||||||
EXCLUDE_METHODS = [ [ "Context.cs", "public Expr MkNumeral(ulong" ],
|
|
||||||
[ "Context.cs", "public Expr MkNumeral(uint" ],
|
|
||||||
[ "Context.cs", "public RatNum MkReal(ulong" ],
|
|
||||||
[ "Context.cs", "public RatNum MkReal(uint" ],
|
|
||||||
[ "Context.cs", "public IntNum MkInt(ulong" ],
|
|
||||||
[ "Context.cs", "public IntNum MkInt(uint" ],
|
|
||||||
[ "Context.cs", "public BitVecNum MkBV(ulong" ],
|
|
||||||
[ "Context.cs", "public BitVecNum MkBV(uint" ],
|
|
||||||
]
|
|
||||||
|
|
||||||
ENUMS = []
|
|
||||||
|
|
||||||
def mk_java_bindings():
|
|
||||||
print "Generating Java bindings (from C# bindings in " + CS + ")..."
|
|
||||||
print "Finding enumerations in " + ENUMS_FILE + "..."
|
|
||||||
find_enums(ENUMS_FILE)
|
|
||||||
for root, dirs, files in os.walk(CS):
|
|
||||||
for fn in files:
|
|
||||||
if not fn in EXCLUDE and fn.endswith(EXT):
|
|
||||||
translate(fn)
|
|
||||||
|
|
||||||
def subst_getters(s, getters):
|
|
||||||
for g in getters:
|
|
||||||
s = s.replace(g, g + "()")
|
|
||||||
|
|
||||||
def type_replace(s):
|
|
||||||
s = s.replace(" bool", " boolean")
|
|
||||||
s = s.replace("(bool", "(boolean")
|
|
||||||
s = s.replace("uint", "int")
|
|
||||||
s = s.replace("ulong", "long")
|
|
||||||
s = s.replace("string", "String")
|
|
||||||
s = s.replace("IntPtr", "long")
|
|
||||||
s = s.replace("Dictionary<", "Map<")
|
|
||||||
s = s.replace("UInt64", "long")
|
|
||||||
s = s.replace("Int64", "long")
|
|
||||||
s = s.replace("List<long>", "LinkedList<Long>")
|
|
||||||
s = s.replace("System.Exception", "Exception")
|
|
||||||
return s
|
|
||||||
|
|
||||||
def rename_native(s):
|
|
||||||
while s.find("Native.Z3") != -1:
|
|
||||||
i0 = s.find("Native.Z3")
|
|
||||||
i1 = s.find("(", i0)
|
|
||||||
c0 = s[:i0]
|
|
||||||
c1 = s[i0:i1]
|
|
||||||
c1 = c1.replace("Native.Z3_", "Native.")
|
|
||||||
c2 = s[i1:]
|
|
||||||
lc_callback = lambda pat: pat.group("id").upper()
|
|
||||||
c1 = re.sub("_(?P<id>\w)", lc_callback, c1)
|
|
||||||
s = c0 + c1 + c2
|
|
||||||
return s
|
|
||||||
|
|
||||||
def find_enums(fn):
|
|
||||||
for line in fileinput.input(os.path.join(CS, fn)):
|
|
||||||
s = string.rstrip(string.lstrip(line))
|
|
||||||
if s.startswith("public enum"):
|
|
||||||
ENUMS.append(s.split(" ")[2])
|
|
||||||
|
|
||||||
|
|
||||||
def enum_replace(line):
|
|
||||||
for e in ENUMS:
|
|
||||||
if line.find("case") != -1:
|
|
||||||
line = line.replace(e + ".", "")
|
|
||||||
elif line.find("== (int)") != -1 or line.find("!= (int)") != -1:
|
|
||||||
line = re.sub("\(int\)" + e + "\.(?P<id>[A-Z0-9_]*)", e + ".\g<id>.toInt()", line)
|
|
||||||
elif line.find("==") != -1 or line.find("!=") != -1:
|
|
||||||
line = re.sub(e + "\.(?P<id>[A-Z0-9_]*)", e + ".\g<id>", line)
|
|
||||||
else:
|
|
||||||
# line = re.sub("\(\(" + e + "\)(?P<rest>.*\(.*)\)", "(" + e + ".values()[\g<rest>])", line)
|
|
||||||
line = re.sub("\(" + e + "\)(?P<rest>.*\(.*\))", e + ".fromInt(\g<rest>)", line)
|
|
||||||
return line
|
|
||||||
|
|
||||||
def replace_generals(a):
|
|
||||||
a = re.sub(" NativeObject", " NativeObject()", a)
|
|
||||||
a = re.sub("\.NativeObject", ".NativeObject()", a)
|
|
||||||
a = re.sub("(?P<h>[\.\(])Id", "\g<h>Id()", a)
|
|
||||||
a = a.replace("(Context ==", "(Context() ==")
|
|
||||||
a = a.replace("(Context,", "(Context(),")
|
|
||||||
a = a.replace("Context.", "Context().")
|
|
||||||
a = a.replace(".nCtx", ".nCtx()")
|
|
||||||
a = a.replace("(nCtx", "(nCtx()")
|
|
||||||
a = re.sub("Context\(\).(?P<id>[^_]*)_DRQ", "Context().\g<id>_DRQ()", a)
|
|
||||||
a = re.sub("ASTKind", "ASTKind()", a)
|
|
||||||
a = re.sub("IsExpr(?P<f>[ ;])", "IsExpr()\g<f>", a)
|
|
||||||
a = re.sub("IsNumeral(?P<f>[ ;])", "IsNumeral()\g<f>", a)
|
|
||||||
a = re.sub("IsInt(?P<f>[ ;])", "IsInt()\g<f>", a)
|
|
||||||
a = re.sub("IsReal(?P<f>[ ;])", "IsReal()\g<f>", a)
|
|
||||||
a = re.sub("IsVar(?P<f>[ ;\)])", "IsVar()\g<f>", a)
|
|
||||||
a = re.sub("FuncDecl.DeclKind", "FuncDecl().DeclKind()", a)
|
|
||||||
a = re.sub("FuncDecl.DomainSize", "FuncDecl().DomainSize()", a)
|
|
||||||
a = re.sub("(?P<h>[=&]) Num(?P<id>[a-zA-Z]*)", "\g<h> Num\g<id>()", a)
|
|
||||||
a = re.sub("= Denominator", "= Denominator()", a)
|
|
||||||
a = re.sub(", BoolSort(?P<f>[\)\.])", ", BoolSort()\g<f>", a)
|
|
||||||
a = re.sub(", RealSort(?P<f>[\)\.])", ", RealSort()\g<f>", a)
|
|
||||||
a = re.sub(", IntSort(?P<f>[\)\.])", ", IntSort()\g<f>", a)
|
|
||||||
a = a.replace("? 1 : 0", "? true : false")
|
|
||||||
if a.find("Native.") != -1 and a.find("!= 0") != -1:
|
|
||||||
a = a.replace("!= 0", "")
|
|
||||||
if a.find("Native.") != -1 and a.find("== 0") != -1:
|
|
||||||
a = a.replace("== 0", "^ true")
|
|
||||||
return a
|
|
||||||
|
|
||||||
def translate(filename):
|
|
||||||
tgtfn = OUTDIR + filename.replace(EXT, ".java")
|
|
||||||
print "Translating " + filename + " to " + tgtfn
|
|
||||||
tgt = open(tgtfn, "w")
|
|
||||||
in_header = 0
|
|
||||||
in_class = 0
|
|
||||||
in_static_class = 0
|
|
||||||
in_javadoc = 0
|
|
||||||
lastindent = 0
|
|
||||||
skip_brace = 0
|
|
||||||
in_getter = ""
|
|
||||||
in_getter_type = ""
|
|
||||||
in_unsupported = 0
|
|
||||||
getters = []
|
|
||||||
in_bracket_op = 0
|
|
||||||
in_getter_get = 0
|
|
||||||
in_getter_set = 0
|
|
||||||
had_ulong_res = 0
|
|
||||||
in_enum = 0
|
|
||||||
missing_foreach_brace = 0
|
|
||||||
foreach_opened_brace = 0
|
|
||||||
for line in fileinput.input(os.path.join(CS, filename)):
|
|
||||||
s = string.rstrip(string.lstrip(line))
|
|
||||||
if in_javadoc:
|
|
||||||
if s.startswith("///"):
|
|
||||||
lastindent = line.find(s);
|
|
||||||
if s.startswith("/// </summary>"):
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
a = line
|
|
||||||
a = a.replace("<c>", "<code>")
|
|
||||||
a = a.replace("</c>", "</code>")
|
|
||||||
a = a.replace("///"," *")
|
|
||||||
a = a.replace("<returns>","@return ")
|
|
||||||
a = a.replace("</returns>","")
|
|
||||||
tgt.write(a)
|
|
||||||
else:
|
|
||||||
t = ""
|
|
||||||
for i in range(0, lastindent):
|
|
||||||
t += " "
|
|
||||||
tgt.write(t + " **/\n")
|
|
||||||
in_javadoc = 0
|
|
||||||
|
|
||||||
# for i in range(0, len(EXCLUDE_METHODS)):
|
|
||||||
# if filename == EXCLUDE_METHODS[i][0] and s.startswith(EXCLUDE_METHODS[i][1]):
|
|
||||||
# tgt.write(t + "/* Not translated because it would translate to a function with clashing types. */\n")
|
|
||||||
# in_unsupported = 1
|
|
||||||
# break
|
|
||||||
|
|
||||||
|
|
||||||
if in_unsupported:
|
|
||||||
if s == "}":
|
|
||||||
in_unsupported = 0
|
|
||||||
elif not in_javadoc:
|
|
||||||
if not in_header and s.find("/*++") != -1:
|
|
||||||
in_header = 1
|
|
||||||
tgt.write("/**\n")
|
|
||||||
elif in_header and s.startswith("--*/"):
|
|
||||||
in_header = 0
|
|
||||||
tgt.write(" * This file was automatically generated from " + filename + " \n")
|
|
||||||
tgt.write(" **/\n")
|
|
||||||
tgt.write("\npackage com.Microsoft.Z3;\n\n")
|
|
||||||
tgt.write("import java.math.BigInteger;\n")
|
|
||||||
tgt.write("import java.util.*;\n")
|
|
||||||
tgt.write("import java.lang.Exception;\n")
|
|
||||||
tgt.write("import com.Microsoft.Z3.Enumerations.*;\n")
|
|
||||||
elif in_header == 1:
|
|
||||||
# tgt.write(" * " + line.replace(filename, tgtfn))
|
|
||||||
pass
|
|
||||||
elif s.startswith("using"):
|
|
||||||
if s.find("System.Diagnostics.Contracts") == -1:
|
|
||||||
tgt.write("/* " + s + " */\n")
|
|
||||||
elif s.startswith("namespace"):
|
|
||||||
pass
|
|
||||||
elif s.startswith("public") and s.find("operator") != -1 and (s.find("==") != -1 or s.find("!=") != -1):
|
|
||||||
t = ""
|
|
||||||
for i in range(0, line.find(s)+1):
|
|
||||||
t += " "
|
|
||||||
tgt.write(t + "/* Overloaded operators are not translated. */\n")
|
|
||||||
in_unsupported = 1
|
|
||||||
elif s.startswith("public enum"):
|
|
||||||
tgt.write(line.replace("enum", "class"))
|
|
||||||
in_enum = 1
|
|
||||||
elif in_enum == 1:
|
|
||||||
if s == "}":
|
|
||||||
tgt.write(line)
|
|
||||||
in_enum = 0
|
|
||||||
else:
|
|
||||||
line = re.sub("(?P<id>.*)\W*=\W*(?P<val>[^\n,])", "public static final int \g<id> = \g<val>;", line)
|
|
||||||
tgt.write(line.replace(",",""))
|
|
||||||
elif s.startswith("public class") or s.startswith("internal class") or s.startswith("internal abstract class"):
|
|
||||||
a = line.replace(":", "extends").replace("internal ", "")
|
|
||||||
a = a.replace(", IComparable", "")
|
|
||||||
a = type_replace(a)
|
|
||||||
tgt.write(a)
|
|
||||||
in_class = 1
|
|
||||||
in_static_class = 0
|
|
||||||
elif s.startswith("public static class") or s.startswith("abstract class"):
|
|
||||||
tgt.write(line.replace(":", "extends").replace("static", "final"))
|
|
||||||
in_class = 1
|
|
||||||
in_static_class = 1
|
|
||||||
elif s.startswith("/// <summary>"):
|
|
||||||
tgt.write(line.replace("/// <summary>", "/**"))
|
|
||||||
in_javadoc = 1
|
|
||||||
elif skip_brace and s == "{":
|
|
||||||
skip_brace = 0
|
|
||||||
elif ((s.find("public") != -1 or s.find("protected") != -1) and s.find("class") == -1 and s.find("event") == -1 and s.find("(") == -1) or s.startswith("internal virtual IntPtr NativeObject") or s.startswith("internal Context Context"):
|
|
||||||
if (s.startswith("new")):
|
|
||||||
s = s[3:]
|
|
||||||
s = s.replace("internal virtual", "")
|
|
||||||
s = s.replace("internal", "")
|
|
||||||
tokens = s.split(" ")
|
|
||||||
# print "TOKENS: " + str(len(tokens))
|
|
||||||
if len(tokens) == 3:
|
|
||||||
in_getter = tokens[2]
|
|
||||||
in_getter_type = type_replace((tokens[0] + " " + tokens[1]))
|
|
||||||
if in_static_class:
|
|
||||||
in_getter_type = in_getter_type.replace("static", "")
|
|
||||||
lastindent = line.find(s)
|
|
||||||
skip_brace = 1
|
|
||||||
getters.append(in_getter)
|
|
||||||
elif len(tokens) == 4:
|
|
||||||
if tokens[2].startswith("this["):
|
|
||||||
in_bracket_op = 1
|
|
||||||
in_getter = type_replace(tokens[2]).replace("this[", "get(")
|
|
||||||
in_getter += " " + tokens[3].replace("]", ")")
|
|
||||||
in_getter_type = type_replace(tokens[0] + " " + tokens[1])
|
|
||||||
else:
|
|
||||||
in_getter = tokens[3]
|
|
||||||
in_getter_type = type_replace(tokens[0] + " " + tokens[1] + " " + tokens[2])
|
|
||||||
if in_static_class:
|
|
||||||
in_getter_type = in_getter_type.replace("static", "")
|
|
||||||
lastindent = line.find(s)
|
|
||||||
skip_brace = 1
|
|
||||||
getters.append(in_getter)
|
|
||||||
else:
|
|
||||||
in_getter = tokens[2]
|
|
||||||
in_getter_type = type_replace(tokens[0] + " " + tokens[1])
|
|
||||||
if tokens[2].startswith("this["):
|
|
||||||
lastindent = line.find(s)
|
|
||||||
t = ""
|
|
||||||
for i in range(0, lastindent): t += " "
|
|
||||||
tgt.write(t + "/* operator this[] not translated */\n ")
|
|
||||||
in_unsupported = 1
|
|
||||||
else:
|
|
||||||
if in_static_class:
|
|
||||||
in_getter_type = in_getter_type.replace("static", "")
|
|
||||||
rest = s[s.find("get ") + 4:-1]
|
|
||||||
subst_getters(rest, getters)
|
|
||||||
rest = type_replace(rest)
|
|
||||||
rest = rename_native(rest)
|
|
||||||
rest = replace_generals(rest)
|
|
||||||
rest = enum_replace(rest)
|
|
||||||
t = ""
|
|
||||||
for i in range(0, lastindent):
|
|
||||||
t += " "
|
|
||||||
tgt.write(t + in_getter_type + " " + in_getter + "() " + rest + "\n")
|
|
||||||
if rest.find("}") == -1:
|
|
||||||
in_getter_get = 1
|
|
||||||
else:
|
|
||||||
getters.append(in_getter)
|
|
||||||
in_getter = ""
|
|
||||||
in_getter_type = ""
|
|
||||||
print "ACC: " + s + " --> " + in_getter
|
|
||||||
elif s.find("{ get {") != -1:
|
|
||||||
line = type_replace(line)
|
|
||||||
line = line.replace("internal ", "")
|
|
||||||
d = line[0:line.find("{ get")]
|
|
||||||
rest = line[line.find("{ get")+5:]
|
|
||||||
rest = rest.replace("} }", "}")
|
|
||||||
rest = re.sub("Contract.\w+\([\s\S]*\);", "", rest)
|
|
||||||
subst_getters(rest, getters)
|
|
||||||
rest = rename_native(rest)
|
|
||||||
rest = replace_generals(rest)
|
|
||||||
rest = enum_replace(rest)
|
|
||||||
if in_bracket_op:
|
|
||||||
tgt.write(d + rest)
|
|
||||||
else:
|
|
||||||
tgt.write(d + "()" + rest)
|
|
||||||
print "ACC: " + s + " --> " + in_getter
|
|
||||||
elif in_getter != "" and s.startswith("get"):
|
|
||||||
t = ""
|
|
||||||
for i in range(0, lastindent):
|
|
||||||
t += " "
|
|
||||||
if len(s) > 3: rest = s[3:]
|
|
||||||
else: rest = ""
|
|
||||||
subst_getters(rest, getters)
|
|
||||||
rest = type_replace(rest)
|
|
||||||
rest = rename_native(rest)
|
|
||||||
rest = replace_generals(rest)
|
|
||||||
rest = enum_replace(rest)
|
|
||||||
if in_bracket_op:
|
|
||||||
tgt.write(t + in_getter_type + " " + in_getter + " " + rest + "\n")
|
|
||||||
else:
|
|
||||||
tgt.write(t + in_getter_type + " " + in_getter + "() " + rest + "\n")
|
|
||||||
if rest.find("}") == -1:
|
|
||||||
in_getter_get = 1
|
|
||||||
elif in_getter != "" and s.startswith("set"):
|
|
||||||
t = ""
|
|
||||||
for i in range(0, lastindent):
|
|
||||||
t += " "
|
|
||||||
if len(s) > 3: rest = type_replace(s[3:])
|
|
||||||
else: rest = ""
|
|
||||||
subst_getters(rest, getters)
|
|
||||||
rest = rest.replace("(Integer)value", "Integer(value)")
|
|
||||||
rest = type_replace(rest)
|
|
||||||
rest = rename_native(rest)
|
|
||||||
rest = replace_generals(rest)
|
|
||||||
rest = enum_replace(rest)
|
|
||||||
ac_acc = in_getter_type[:in_getter_type.find(' ')]
|
|
||||||
ac_type = in_getter_type[in_getter_type.find(' ')+1:]
|
|
||||||
if in_bracket_op:
|
|
||||||
in_getter = in_getter.replace("get", "set").replace(")", "")
|
|
||||||
tgt.write(t + ac_acc + " void " + in_getter + ", " + ac_type + " value) " + rest + "\n")
|
|
||||||
else:
|
|
||||||
tgt.write(t + ac_acc + " void set" + in_getter + "(" + ac_type + " value) " + rest + "\n")
|
|
||||||
if rest.find("}") == -1:
|
|
||||||
in_getter_set = 1
|
|
||||||
elif in_getter != "" and in_getter_get == 1 and s == "}":
|
|
||||||
tgt.write(line)
|
|
||||||
in_getter_get = 0
|
|
||||||
elif in_getter != "" and in_getter_set == 1 and s == "}":
|
|
||||||
tgt.write(line)
|
|
||||||
in_getter_set = 0
|
|
||||||
elif in_getter != "" and in_getter_get == 0 and in_getter_set == 0 and s == "}":
|
|
||||||
in_getter = ""
|
|
||||||
in_getter_type == ""
|
|
||||||
in_bracket_op = 0
|
|
||||||
skip_brace = 0
|
|
||||||
elif s.startswith("uint ") and s.find("=") == -1:
|
|
||||||
line = line.replace("uint", "Integer", line)
|
|
||||||
line = re.sub("(?P<n>\w+)(?P<c>[,;])", "\g<n>\g<c>", line)
|
|
||||||
tgt.write(line);
|
|
||||||
elif (not in_class and (s.startswith("{") or s.startswith("}"))) or s.startswith("[") or s.startswith("#"):
|
|
||||||
# tgt.write("// Skipping: " + s)
|
|
||||||
pass
|
|
||||||
elif line == "}\n":
|
|
||||||
pass
|
|
||||||
else:
|
|
||||||
# indent = line.find(s)
|
|
||||||
# tgt.write("// LINE: " + line)
|
|
||||||
if line.find(": base") != -1:
|
|
||||||
line = re.sub(": base\((?P<p>[^\{]*)\)", "{ super(\g<p>);", line)
|
|
||||||
line = line[4:]
|
|
||||||
obraces = line.count("{")
|
|
||||||
cbraces = line.count("}")
|
|
||||||
mbraces = obraces - cbraces
|
|
||||||
# tgt.write("// obraces = " + str(obraces) + "\n")
|
|
||||||
if obraces == 1:
|
|
||||||
skip_brace = 1
|
|
||||||
else:
|
|
||||||
for i in range(0, mbraces):
|
|
||||||
line = line.replace("\n", "}\n")
|
|
||||||
if (s.find("public") != -1 or s.find("protected") != -1 or s.find("internal") != -1) and s.find("(") != -1:
|
|
||||||
line = re.sub(" = [\w.]+(?P<d>[,;\)])", "\g<d>", line)
|
|
||||||
a = type_replace(line)
|
|
||||||
a = enum_replace(a)
|
|
||||||
a = re.sub("(?P<d>[\(, ])params ", "\g<d>", a)
|
|
||||||
a = a.replace("base.", "super.")
|
|
||||||
a = re.sub("Contract.\w+\([\s\S]*\);", "", a)
|
|
||||||
a = rename_native(a)
|
|
||||||
a = re.sub("~\w+\(\)", "protected void finalize()", a)
|
|
||||||
|
|
||||||
if missing_foreach_brace == 1:
|
|
||||||
# a = a.replace("\n", " // checked " + str(foreach_opened_brace) + "\n")
|
|
||||||
if foreach_opened_brace == 0 and a.find("{") != -1:
|
|
||||||
foreach_opened_brace = 1
|
|
||||||
elif foreach_opened_brace == 0 and a.find("}") == -1:
|
|
||||||
a = a.replace("\n", "}}\n")
|
|
||||||
foreach_opened_brace = 0
|
|
||||||
missing_foreach_brace = 0
|
|
||||||
elif foreach_opened_brace == 1 and a.find("}") != -1:
|
|
||||||
a = a.replace("\n", "}}\n")
|
|
||||||
foreach_opened_brace = 0
|
|
||||||
missing_foreach_brace = 0
|
|
||||||
|
|
||||||
# if a.find("foreach") != -1:
|
|
||||||
# missing_foreach_brace = 1
|
|
||||||
# a = re.sub("foreach\s*\((?P<t>[\w <>,]+)\s+(?P<i>\w+)\s+in\s+(?P<w>\w+)\)",
|
|
||||||
# "{ Iterator fe_i = \g<w>.iterator(); while (fe_i.hasNext()) { \g<t> \g<i> = (long)fe_i.next(); ",
|
|
||||||
# a)
|
|
||||||
a = re.sub("foreach\s*\((?P<t>[\w <>,]+)\s+(?P<i>\w+)\s+in\s+(?P<w>\w+)\)",
|
|
||||||
"for (\g<t> \g<i>: \g<w>)",
|
|
||||||
a)
|
|
||||||
if a.find("long o: m_queue") != -1:
|
|
||||||
a = a.replace("long", "Long")
|
|
||||||
a = a.replace("readonly ", "")
|
|
||||||
a = a.replace("const ", "final ")
|
|
||||||
a = a.replace("String ToString", "String toString")
|
|
||||||
a = a.replace(".ToString", ".toString")
|
|
||||||
a = a.replace("internal ", "")
|
|
||||||
a = a.replace("new static", "static")
|
|
||||||
a = a.replace("new public", "public")
|
|
||||||
a = a.replace("override ", "")
|
|
||||||
a = a.replace("virtual ", "")
|
|
||||||
a = a.replace("o as AST", "(AST) o")
|
|
||||||
a = a.replace("o as Sort", "(Sort) o")
|
|
||||||
a = a.replace("other as AST", "(AST) other")
|
|
||||||
a = a.replace("o as FuncDecl", "(FuncDecl) o")
|
|
||||||
a = a.replace("IntPtr obj", "long obj")
|
|
||||||
a = a.replace("IntPtr o", "long o")
|
|
||||||
a = a.replace("new long()", "0")
|
|
||||||
a = a.replace("long.Zero", "0")
|
|
||||||
a = a.replace("object o", "Object o")
|
|
||||||
a = a.replace("object other", "Object other")
|
|
||||||
a = a.replace("IntPtr res = IntPtr.Zero;", "Native.IntPtr res = new Native.IntPtr();")
|
|
||||||
a = a.replace("out res", "res")
|
|
||||||
a = a.replace("GC.ReRegisterForFinalize(m_ctx);", "")
|
|
||||||
a = a.replace("GC.SuppressFinalize(this);", "")
|
|
||||||
a = a.replace(".Length", ".length")
|
|
||||||
a = a.replace("m_queue.Count", "m_queue.size()")
|
|
||||||
a = a.replace("m_queue.Add", "m_queue.add")
|
|
||||||
a = a.replace("m_queue.Clear", "m_queue.clear")
|
|
||||||
a = a.replace("for (long ", "for (int ")
|
|
||||||
a = a.replace("ReferenceEquals(Context, ctx)", "Context() == ctx")
|
|
||||||
a = a.replace("BigInteger.Parse", "new BigInteger")
|
|
||||||
if had_ulong_res == 0 and a.find("ulong res = 0") != -1:
|
|
||||||
a = a.replace("ulong res = 0;", "LongPtr res = new LongPtr();")
|
|
||||||
elif had_ulong_res == 1:
|
|
||||||
a = a.replace("ref res)", "res)")
|
|
||||||
if a.find("return res;") != -1:
|
|
||||||
a = a.replace("return res;", "return res.value;")
|
|
||||||
had_ulong_res = 0
|
|
||||||
a = a.replace("lock (", "synchronized (")
|
|
||||||
if in_static_class:
|
|
||||||
a = a.replace("static", "")
|
|
||||||
a = re.sub("ref (?P<id>\w+)", "\g<id>", a)
|
|
||||||
subst_getters(a, getters)
|
|
||||||
a = re.sub("NativeObject = (?P<rest>.*);", "setNativeObject(\g<rest>);", a)
|
|
||||||
a = replace_generals(a)
|
|
||||||
tgt.write(a)
|
|
||||||
tgt.close()
|
|
||||||
|
|
||||||
mk_java_bindings()
|
|
|
@ -1,5 +1,5 @@
|
||||||
/*++
|
/*++
|
||||||
Copyright (c) 2010 Microsoft Corporation
|
Copyright (c) Microsoft Corporation
|
||||||
|
|
||||||
Module Name:
|
Module Name:
|
||||||
|
|
||||||
|
|
|
@ -38,7 +38,7 @@ Z3 exceptions:
|
||||||
... n = x + y
|
... n = x + y
|
||||||
... except Z3Exception as ex:
|
... except Z3Exception as ex:
|
||||||
... print("failed: %s" % ex)
|
... print("failed: %s" % ex)
|
||||||
failed: 'sort mismatch'
|
failed: sort mismatch
|
||||||
"""
|
"""
|
||||||
from z3core import *
|
from z3core import *
|
||||||
from z3types import *
|
from z3types import *
|
||||||
|
@ -1396,7 +1396,7 @@ def BoolVector(prefix, sz, ctx=None):
|
||||||
return [ Bool('%s__%s' % (prefix, i)) for i in range(sz) ]
|
return [ Bool('%s__%s' % (prefix, i)) for i in range(sz) ]
|
||||||
|
|
||||||
def FreshBool(prefix='b', ctx=None):
|
def FreshBool(prefix='b', ctx=None):
|
||||||
"""Return a fresh Bolean constant in the given context using the given prefix.
|
"""Return a fresh Boolean constant in the given context using the given prefix.
|
||||||
|
|
||||||
If `ctx=None`, then the global context is used.
|
If `ctx=None`, then the global context is used.
|
||||||
|
|
||||||
|
@ -6162,7 +6162,7 @@ class Fixedpoint(Z3PPObject):
|
||||||
Z3_fixedpoint_add_rule(self.ctx.ref(), self.fixedpoint, head.as_ast(), name)
|
Z3_fixedpoint_add_rule(self.ctx.ref(), self.fixedpoint, head.as_ast(), name)
|
||||||
else:
|
else:
|
||||||
body = _get_args(body)
|
body = _get_args(body)
|
||||||
f = self.abstract(Implies(And(body),head))
|
f = self.abstract(Implies(And(body,self.ctx),head))
|
||||||
Z3_fixedpoint_add_rule(self.ctx.ref(), self.fixedpoint, f.as_ast(), name)
|
Z3_fixedpoint_add_rule(self.ctx.ref(), self.fixedpoint, f.as_ast(), name)
|
||||||
|
|
||||||
def rule(self, head, body = None, name = None):
|
def rule(self, head, body = None, name = None):
|
||||||
|
@ -6190,7 +6190,7 @@ class Fixedpoint(Z3PPObject):
|
||||||
if sz == 1:
|
if sz == 1:
|
||||||
query = query[0]
|
query = query[0]
|
||||||
else:
|
else:
|
||||||
query = And(query)
|
query = And(query, self.ctx)
|
||||||
query = self.abstract(query, False)
|
query = self.abstract(query, False)
|
||||||
r = Z3_fixedpoint_query(self.ctx.ref(), self.fixedpoint, query.as_ast())
|
r = Z3_fixedpoint_query(self.ctx.ref(), self.fixedpoint, query.as_ast())
|
||||||
return CheckSatResult(r)
|
return CheckSatResult(r)
|
||||||
|
@ -6209,7 +6209,7 @@ class Fixedpoint(Z3PPObject):
|
||||||
name = ""
|
name = ""
|
||||||
name = to_symbol(name, self.ctx)
|
name = to_symbol(name, self.ctx)
|
||||||
body = _get_args(body)
|
body = _get_args(body)
|
||||||
f = self.abstract(Implies(And(body),head))
|
f = self.abstract(Implies(And(body, self.ctx),head))
|
||||||
Z3_fixedpoint_update_rule(self.ctx.ref(), self.fixedpoint, f.as_ast(), name)
|
Z3_fixedpoint_update_rule(self.ctx.ref(), self.fixedpoint, f.as_ast(), name)
|
||||||
|
|
||||||
def get_answer(self):
|
def get_answer(self):
|
||||||
|
|
|
@ -23,7 +23,19 @@ Notes:
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
/**
|
||||||
|
\defgroup capi C API
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
@name Algebraic Numbers API
|
||||||
|
*/
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Return Z3_TRUE if \c can be used as value in the Z3 real algebraic
|
\brief Return Z3_TRUE if \c can be used as value in the Z3 real algebraic
|
||||||
|
@ -218,6 +230,8 @@ extern "C" {
|
||||||
*/
|
*/
|
||||||
int Z3_API Z3_algebraic_eval(__in Z3_context c, __in Z3_ast p, __in unsigned n, __in Z3_ast a[]);
|
int Z3_API Z3_algebraic_eval(__in Z3_context c, __in Z3_ast p, __in unsigned n, __in Z3_ast a[]);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
};
|
};
|
||||||
|
|
|
@ -773,7 +773,6 @@ typedef enum
|
||||||
|
|
||||||
The premises of the rules is a sequence of clauses.
|
The premises of the rules is a sequence of clauses.
|
||||||
The first clause argument is the main clause of the rule.
|
The first clause argument is the main clause of the rule.
|
||||||
One literal from the second, third, .. clause is resolved
|
|
||||||
with a literal from the first (main) clause.
|
with a literal from the first (main) clause.
|
||||||
|
|
||||||
Premises of the rules are of the form
|
Premises of the rules are of the form
|
||||||
|
@ -1159,8 +1158,8 @@ typedef enum {
|
||||||
- Z3_FILE_ACCESS_ERRROR: A file could not be accessed.
|
- Z3_FILE_ACCESS_ERRROR: A file could not be accessed.
|
||||||
- Z3_INVALID_USAGE: API call is invalid in the current state.
|
- Z3_INVALID_USAGE: API call is invalid in the current state.
|
||||||
- Z3_INTERNAL_FATAL: An error internal to Z3 occurred.
|
- Z3_INTERNAL_FATAL: An error internal to Z3 occurred.
|
||||||
- Z3_DEC_REF_ERROR: Trying to decrement the reference counter of an AST that was deleted or the reference counter was not initialized\mlonly.\endmlonly\conly with #Z3_inc_ref.
|
- Z3_DEC_REF_ERROR: Trying to decrement the reference counter of an AST that was deleted or the reference counter was not initialized \mlonly.\endmlonly \conly with #Z3_inc_ref.
|
||||||
- Z3_EXCEPTION: Internal Z3 exception. Additional details can be retrieved using \mlonly #Z3_get_error_msg. \endmlonly \conly #Z3_get_error_msg_ex.
|
- Z3_EXCEPTION: Internal Z3 exception. Additional details can be retrieved using #Z3_get_error_msg.
|
||||||
*/
|
*/
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
|
@ -1304,8 +1303,6 @@ extern "C" {
|
||||||
|
|
||||||
\sa Z3_global_param_set
|
\sa Z3_global_param_set
|
||||||
|
|
||||||
The caller must invoke #Z3_global_param_del_value to delete the value returned at \c param_value.
|
|
||||||
|
|
||||||
\remark This function cannot be invoked simultaneously from different threads without synchronization.
|
\remark This function cannot be invoked simultaneously from different threads without synchronization.
|
||||||
The result string stored in param_value is stored in shared location.
|
The result string stored in param_value is stored in shared location.
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,14 @@ extern "C" {
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@name Interpolation
|
\defgroup capi C API
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
@name Interpolation API
|
||||||
*/
|
*/
|
||||||
/*@{*/
|
/*@{*/
|
||||||
|
|
||||||
|
@ -184,6 +191,8 @@ extern "C" {
|
||||||
\param cnsts Returns sequence of formulas (do not free)
|
\param cnsts Returns sequence of formulas (do not free)
|
||||||
\param parents Returns the parents vector (or NULL for sequence)
|
\param parents Returns the parents vector (or NULL for sequence)
|
||||||
\param error Returns an error message in case of failure (do not free the string)
|
\param error Returns an error message in case of failure (do not free the string)
|
||||||
|
\param num_theory Number of theory terms
|
||||||
|
\param theory Theory terms
|
||||||
|
|
||||||
Returns true on success.
|
Returns true on success.
|
||||||
|
|
||||||
|
@ -232,6 +241,8 @@ extern "C" {
|
||||||
\param parents The parents vector (or NULL for sequence)
|
\param parents The parents vector (or NULL for sequence)
|
||||||
\param interps The interpolant to check
|
\param interps The interpolant to check
|
||||||
\param error Returns an error message if interpolant incorrect (do not free the string)
|
\param error Returns an error message if interpolant incorrect (do not free the string)
|
||||||
|
\param num_theory Number of theory terms
|
||||||
|
\param theory Theory terms
|
||||||
|
|
||||||
Return value is Z3_L_TRUE if interpolant is verified, Z3_L_FALSE if
|
Return value is Z3_L_TRUE if interpolant is verified, Z3_L_FALSE if
|
||||||
incorrect, and Z3_L_UNDEF if unknown.
|
incorrect, and Z3_L_UNDEF if unknown.
|
||||||
|
@ -258,6 +269,8 @@ extern "C" {
|
||||||
\param cnsts Array of constraints
|
\param cnsts Array of constraints
|
||||||
\param parents The parents vector (or NULL for sequence)
|
\param parents The parents vector (or NULL for sequence)
|
||||||
\param filename The file name to write
|
\param filename The file name to write
|
||||||
|
\param num_theory Number of theory terms
|
||||||
|
\param theory Theory terms
|
||||||
|
|
||||||
def_API('Z3_write_interpolation_problem', VOID, (_in(CONTEXT), _in(UINT), _in_array(1, AST), _in_array(1, UINT), _in(STRING), _in(UINT), _in_array(5, AST)))
|
def_API('Z3_write_interpolation_problem', VOID, (_in(CONTEXT), _in(UINT), _in_array(1, AST), _in_array(1, UINT), _in(STRING), _in(UINT), _in_array(5, AST)))
|
||||||
*/
|
*/
|
||||||
|
@ -270,6 +283,9 @@ extern "C" {
|
||||||
__in unsigned num_theory,
|
__in unsigned num_theory,
|
||||||
__in_ecount(num_theory) Z3_ast theory[]);
|
__in_ecount(num_theory) Z3_ast theory[]);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
};
|
};
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
|
@ -24,6 +24,19 @@ Notes:
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
/**
|
||||||
|
\defgroup capi C API
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
@name Polynomials API
|
||||||
|
*/
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Return the nonzero subresultants of \c p and \c q with respect to the "variable" \c x.
|
\brief Return the nonzero subresultants of \c p and \c q with respect to the "variable" \c x.
|
||||||
|
|
||||||
|
@ -38,6 +51,9 @@ extern "C" {
|
||||||
Z3_ast_vector Z3_API Z3_polynomial_subresultants(__in Z3_context c, __in Z3_ast p, __in Z3_ast q, __in Z3_ast x);
|
Z3_ast_vector Z3_API Z3_polynomial_subresultants(__in Z3_context c, __in Z3_ast p, __in Z3_ast q, __in Z3_ast x);
|
||||||
|
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
};
|
};
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
|
@ -25,6 +25,18 @@ Notes:
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
extern "C" {
|
extern "C" {
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
||||||
|
/**
|
||||||
|
\defgroup capi C API
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
@name Real Closed Fields API
|
||||||
|
*/
|
||||||
|
/*@{*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Delete a RCF numeral created using the RCF API.
|
\brief Delete a RCF numeral created using the RCF API.
|
||||||
|
@ -192,6 +204,9 @@ extern "C" {
|
||||||
*/
|
*/
|
||||||
void Z3_API Z3_rcf_get_numerator_denominator(__in Z3_context c, __in Z3_rcf_num a, __out Z3_rcf_num * n, __out Z3_rcf_num * d);
|
void Z3_API Z3_rcf_get_numerator_denominator(__in Z3_context c, __in Z3_rcf_num a, __out Z3_rcf_num * n, __out Z3_rcf_num * d);
|
||||||
|
|
||||||
|
/*@}*/
|
||||||
|
/*@}*/
|
||||||
|
|
||||||
#ifdef __cplusplus
|
#ifdef __cplusplus
|
||||||
};
|
};
|
||||||
#endif // __cplusplus
|
#endif // __cplusplus
|
||||||
|
|
|
@ -40,11 +40,10 @@ void float_decl_plugin::set_manager(ast_manager * m, family_id id) {
|
||||||
SASSERT(m_int_sort != 0); // arith_decl_plugin must be installed before float_decl_plugin.
|
SASSERT(m_int_sort != 0); // arith_decl_plugin must be installed before float_decl_plugin.
|
||||||
m_manager->inc_ref(m_int_sort);
|
m_manager->inc_ref(m_int_sort);
|
||||||
|
|
||||||
if (m_manager->has_plugin(symbol("bv"))) {
|
// BV is not optional anymore.
|
||||||
// bv plugin is optional, so m_bv_plugin may be 0
|
SASSERT(m_manager->has_plugin(symbol("bv")));
|
||||||
m_bv_fid = m_manager->mk_family_id("bv");
|
m_bv_fid = m_manager->mk_family_id("bv");
|
||||||
m_bv_plugin = static_cast<bv_decl_plugin*>(m_manager->get_plugin(m_bv_fid));
|
m_bv_plugin = static_cast<bv_decl_plugin*>(m_manager->get_plugin(m_bv_fid));
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
float_decl_plugin::~float_decl_plugin() {
|
float_decl_plugin::~float_decl_plugin() {
|
||||||
|
@ -103,6 +102,18 @@ bool float_decl_plugin::is_value(expr * n, mpf & val) {
|
||||||
m_fm.mk_nan(ebits, sbits, val);
|
m_fm.mk_nan(ebits, sbits, val);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
else if (is_app_of(n, m_family_id, OP_FLOAT_PLUS_ZERO)) {
|
||||||
|
unsigned ebits = to_app(n)->get_decl()->get_range()->get_parameter(0).get_int();
|
||||||
|
unsigned sbits = to_app(n)->get_decl()->get_range()->get_parameter(1).get_int();
|
||||||
|
m_fm.mk_pzero(ebits, sbits, val);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
else if (is_app_of(n, m_family_id, OP_FLOAT_MINUS_ZERO)) {
|
||||||
|
unsigned ebits = to_app(n)->get_decl()->get_range()->get_parameter(0).get_int();
|
||||||
|
unsigned sbits = to_app(n)->get_decl()->get_range()->get_parameter(1).get_int();
|
||||||
|
m_fm.mk_nzero(ebits, sbits, val);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,7 +167,7 @@ sort * float_decl_plugin::mk_float_sort(unsigned ebits, unsigned sbits) {
|
||||||
parameter ps[2] = { p1, p2 };
|
parameter ps[2] = { p1, p2 };
|
||||||
sort_size sz;
|
sort_size sz;
|
||||||
sz = sort_size::mk_very_big(); // TODO: refine
|
sz = sort_size::mk_very_big(); // TODO: refine
|
||||||
return m_manager->mk_sort(symbol("FP"), sort_info(m_family_id, FLOAT_SORT, sz, 2, ps));
|
return m_manager->mk_sort(symbol("FloatingPoint"), sort_info(m_family_id, FLOAT_SORT, sz, 2, ps));
|
||||||
}
|
}
|
||||||
|
|
||||||
sort * float_decl_plugin::mk_rm_sort() {
|
sort * float_decl_plugin::mk_rm_sort() {
|
||||||
|
@ -176,6 +187,14 @@ sort * float_decl_plugin::mk_sort(decl_kind k, unsigned num_parameters, paramete
|
||||||
return mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
return mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
||||||
case ROUNDING_MODE_SORT:
|
case ROUNDING_MODE_SORT:
|
||||||
return mk_rm_sort();
|
return mk_rm_sort();
|
||||||
|
case FLOAT16_SORT:
|
||||||
|
return mk_float_sort(5, 11);
|
||||||
|
case FLOAT32_SORT:
|
||||||
|
return mk_float_sort(8, 24);
|
||||||
|
case FLOAT64_SORT:
|
||||||
|
return mk_float_sort(11, 53);
|
||||||
|
case FLOAT128_SORT:
|
||||||
|
return mk_float_sort(15, 133);
|
||||||
default:
|
default:
|
||||||
m_manager->raise_exception("unknown floating point theory sort");
|
m_manager->raise_exception("unknown floating point theory sort");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -229,17 +248,18 @@ func_decl * float_decl_plugin::mk_float_const_decl(decl_kind k, unsigned num_par
|
||||||
unsigned ebits = s->get_parameter(0).get_int();
|
unsigned ebits = s->get_parameter(0).get_int();
|
||||||
unsigned sbits = s->get_parameter(1).get_int();
|
unsigned sbits = s->get_parameter(1).get_int();
|
||||||
scoped_mpf val(m_fm);
|
scoped_mpf val(m_fm);
|
||||||
if (k == OP_FLOAT_NAN) {
|
|
||||||
m_fm.mk_nan(ebits, sbits, val);
|
switch (k)
|
||||||
|
{
|
||||||
|
case OP_FLOAT_NAN: m_fm.mk_nan(ebits, sbits, val);
|
||||||
SASSERT(m_fm.is_nan(val));
|
SASSERT(m_fm.is_nan(val));
|
||||||
|
break;
|
||||||
|
case OP_FLOAT_MINUS_INF: m_fm.mk_ninf(ebits, sbits, val); break;
|
||||||
|
case OP_FLOAT_PLUS_INF: m_fm.mk_pinf(ebits, sbits, val); break;
|
||||||
|
case OP_FLOAT_MINUS_ZERO: m_fm.mk_nzero(ebits, sbits, val); break;
|
||||||
|
case OP_FLOAT_PLUS_ZERO: m_fm.mk_pzero(ebits, sbits, val); break;
|
||||||
}
|
}
|
||||||
else if (k == OP_FLOAT_MINUS_INF) {
|
|
||||||
m_fm.mk_ninf(ebits, sbits, val);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
SASSERT(k == OP_FLOAT_PLUS_INF);
|
|
||||||
m_fm.mk_pinf(ebits, sbits, val);
|
|
||||||
}
|
|
||||||
return mk_value_decl(val);
|
return mk_value_decl(val);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -248,14 +268,14 @@ func_decl * float_decl_plugin::mk_bin_rel_decl(decl_kind k, unsigned num_paramet
|
||||||
if (arity != 2)
|
if (arity != 2)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point relation");
|
m_manager->raise_exception("invalid number of arguments to floating point relation");
|
||||||
if (domain[0] != domain[1] || !is_float_sort(domain[0]))
|
if (domain[0] != domain[1] || !is_float_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected equal FloatingPoint sorts as arguments");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_EQ: name = "=="; break;
|
case OP_FLOAT_EQ: name = "fp.eq"; break;
|
||||||
case OP_FLOAT_LT: name = "<"; break;
|
case OP_FLOAT_LT: name = "fp.lt"; break;
|
||||||
case OP_FLOAT_GT: name = ">"; break;
|
case OP_FLOAT_GT: name = "fp.gt"; break;
|
||||||
case OP_FLOAT_LE: name = "<="; break;
|
case OP_FLOAT_LE: name = "fp.lte"; break;
|
||||||
case OP_FLOAT_GE: name = ">="; break;
|
case OP_FLOAT_GE: name = "fp.gte"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -270,17 +290,18 @@ func_decl * float_decl_plugin::mk_unary_rel_decl(decl_kind k, unsigned num_param
|
||||||
if (arity != 1)
|
if (arity != 1)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point relation");
|
m_manager->raise_exception("invalid number of arguments to floating point relation");
|
||||||
if (!is_float_sort(domain[0]))
|
if (!is_float_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected argument of FloatingPoint sort");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_IS_ZERO: name = "isZero"; break;
|
case OP_FLOAT_IS_ZERO: name = "fp.isZero"; break;
|
||||||
case OP_FLOAT_IS_NZERO: name = "isNZero"; break;
|
case OP_FLOAT_IS_NZERO: name = "fp.isNZero"; break;
|
||||||
case OP_FLOAT_IS_PZERO: name = "isPZero"; break;
|
case OP_FLOAT_IS_PZERO: name = "fp.isPZero"; break;
|
||||||
case OP_FLOAT_IS_SIGN_MINUS: name = "isSignMinus"; break;
|
case OP_FLOAT_IS_NEGATIVE: name = "fp.isNegative"; break;
|
||||||
case OP_FLOAT_IS_NAN: name = "isNaN"; break;
|
case OP_FLOAT_IS_POSITIVE: name = "fp.isPositive"; break;
|
||||||
case OP_FLOAT_IS_INF: name = "isInfinite"; break;
|
case OP_FLOAT_IS_NAN: name = "fp.isNaN"; break;
|
||||||
case OP_FLOAT_IS_NORMAL: name = "isNormal"; break;
|
case OP_FLOAT_IS_INF: name = "fp.isInfinite"; break;
|
||||||
case OP_FLOAT_IS_SUBNORMAL: name = "isSubnormal"; break;
|
case OP_FLOAT_IS_NORMAL: name = "fp.isNormal"; break;
|
||||||
|
case OP_FLOAT_IS_SUBNORMAL: name = "fp.isSubnormal"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -293,11 +314,11 @@ func_decl * float_decl_plugin::mk_unary_decl(decl_kind k, unsigned num_parameter
|
||||||
if (arity != 1)
|
if (arity != 1)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
||||||
if (!is_float_sort(domain[0]))
|
if (!is_float_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected argument of FloatingPoint sort");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_ABS: name = "abs"; break;
|
case OP_FLOAT_ABS: name = "fp.abs"; break;
|
||||||
case OP_FLOAT_UMINUS: name = "-"; break;
|
case OP_FLOAT_NEG: name = "fp.neg"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -310,12 +331,12 @@ func_decl * float_decl_plugin::mk_binary_decl(decl_kind k, unsigned num_paramete
|
||||||
if (arity != 2)
|
if (arity != 2)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
||||||
if (domain[0] != domain[1] || !is_float_sort(domain[0]))
|
if (domain[0] != domain[1] || !is_float_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected arguments of equal FloatingPoint sorts");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_REM: name = "remainder"; break;
|
case OP_FLOAT_REM: name = "fp.rem"; break;
|
||||||
case OP_FLOAT_MIN: name = "fp.min"; break;
|
case OP_FLOAT_MIN: name = "fp.min"; break;
|
||||||
case OP_FLOAT_MAX: name = "fp.max"; break;
|
case OP_FLOAT_MAX: name = "fp.max"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -327,14 +348,16 @@ func_decl * float_decl_plugin::mk_rm_binary_decl(decl_kind k, unsigned num_param
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (arity != 3)
|
if (arity != 3)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
||||||
if (!is_rm_sort(domain[0]) || domain[1] != domain[2] || !is_float_sort(domain[1]))
|
if (!is_rm_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected first argument of RoundingMode sort");
|
||||||
|
if (domain[1] != domain[2] || !is_float_sort(domain[1]))
|
||||||
|
m_manager->raise_exception("sort mismatch, expected arguments 1 and 2 of equal FloatingPoint sorts");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_ADD: name = "+"; break;
|
case OP_FLOAT_ADD: name = "fp.add"; break;
|
||||||
case OP_FLOAT_SUB: name = "-"; break;
|
case OP_FLOAT_SUB: name = "fp.sub"; break;
|
||||||
case OP_FLOAT_MUL: name = "*"; break;
|
case OP_FLOAT_MUL: name = "fp.mul"; break;
|
||||||
case OP_FLOAT_DIV: name = "/"; break;
|
case OP_FLOAT_DIV: name = "fp.div"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -346,12 +369,14 @@ func_decl * float_decl_plugin::mk_rm_unary_decl(decl_kind k, unsigned num_parame
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (arity != 2)
|
if (arity != 2)
|
||||||
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
m_manager->raise_exception("invalid number of arguments to floating point operator");
|
||||||
if (!is_rm_sort(domain[0]) || !is_float_sort(domain[1]))
|
if (!is_rm_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected RoundingMode as first argument");
|
||||||
|
if (!is_float_sort(domain[1]))
|
||||||
|
m_manager->raise_exception("sort mismatch, expected FloatingPoint as second argument");
|
||||||
symbol name;
|
symbol name;
|
||||||
switch (k) {
|
switch (k) {
|
||||||
case OP_FLOAT_SQRT: name = "squareRoot"; break;
|
case OP_FLOAT_SQRT: name = "fp.sqrt"; break;
|
||||||
case OP_FLOAT_ROUND_TO_INTEGRAL: name = "roundToIntegral"; break;
|
case OP_FLOAT_ROUND_TO_INTEGRAL: name = "fp.roundToIntegral"; break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
break;
|
break;
|
||||||
|
@ -359,13 +384,15 @@ func_decl * float_decl_plugin::mk_rm_unary_decl(decl_kind k, unsigned num_parame
|
||||||
return m_manager->mk_func_decl(name, arity, domain, domain[1], func_decl_info(m_family_id, k));
|
return m_manager->mk_func_decl(name, arity, domain, domain[1], func_decl_info(m_family_id, k));
|
||||||
}
|
}
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_fused_ma(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * float_decl_plugin::mk_fma(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (arity != 4)
|
if (arity != 4)
|
||||||
m_manager->raise_exception("invalid number of arguments to fused_ma operator");
|
m_manager->raise_exception("invalid number of arguments to fused_ma operator");
|
||||||
if (!is_rm_sort(domain[0]) || domain[1] != domain[2] || domain[1] != domain[3] || !is_float_sort(domain[1]))
|
if (!is_rm_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected RoundingMode as first argument");
|
||||||
symbol name("fusedMA");
|
if (domain[1] != domain[2] || domain[1] != domain[3] || !is_float_sort(domain[1]))
|
||||||
|
m_manager->raise_exception("sort mismatch, expected arguments 1,2,3 of equal FloatingPoint sort");
|
||||||
|
symbol name("fp.fma");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, domain[1], func_decl_info(m_family_id, k));
|
return m_manager->mk_func_decl(name, arity, domain, domain[1], func_decl_info(m_family_id, k));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -375,12 +402,13 @@ func_decl * float_decl_plugin::mk_to_float(decl_kind k, unsigned num_parameters,
|
||||||
is_sort_of(domain[0], m_bv_fid, BV_SORT) &&
|
is_sort_of(domain[0], m_bv_fid, BV_SORT) &&
|
||||||
is_sort_of(domain[1], m_bv_fid, BV_SORT) &&
|
is_sort_of(domain[1], m_bv_fid, BV_SORT) &&
|
||||||
is_sort_of(domain[2], m_bv_fid, BV_SORT)) {
|
is_sort_of(domain[2], m_bv_fid, BV_SORT)) {
|
||||||
// When the bv_decl_plugin is installed, then we know how to convert 3 bit-vectors into a float!
|
// 3 BVs -> 1 FP
|
||||||
sort * fp = mk_float_sort(domain[2]->get_parameter(0).get_int(), domain[1]->get_parameter(0).get_int()+1);
|
sort * fp = mk_float_sort(domain[2]->get_parameter(0).get_int(), domain[1]->get_parameter(0).get_int()+1);
|
||||||
symbol name("asFloat");
|
symbol name("fp");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
||||||
}
|
}
|
||||||
else if (m_bv_plugin && arity == 1 && is_sort_of(domain[0], m_bv_fid, BV_SORT)) {
|
else if (m_bv_plugin && arity == 1 && is_sort_of(domain[0], m_bv_fid, BV_SORT)) {
|
||||||
|
// 1 BV -> 1 FP
|
||||||
if (num_parameters != 2)
|
if (num_parameters != 2)
|
||||||
m_manager->raise_exception("invalid number of parameters to to_fp");
|
m_manager->raise_exception("invalid number of parameters to to_fp");
|
||||||
if (!parameters[0].is_int() || !parameters[1].is_int())
|
if (!parameters[0].is_int() || !parameters[1].is_int())
|
||||||
|
@ -389,37 +417,67 @@ func_decl * float_decl_plugin::mk_to_float(decl_kind k, unsigned num_parameters,
|
||||||
int sbits = parameters[1].get_int();
|
int sbits = parameters[1].get_int();
|
||||||
|
|
||||||
sort * fp = mk_float_sort(ebits, sbits);
|
sort * fp = mk_float_sort(ebits, sbits);
|
||||||
symbol name("asFloat");
|
symbol name("to_fp");
|
||||||
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
||||||
|
}
|
||||||
|
else if (m_bv_plugin && arity == 2 &&
|
||||||
|
is_sort_of(domain[0], m_family_id, ROUNDING_MODE_SORT) &&
|
||||||
|
is_sort_of(domain[1], m_bv_fid, BV_SORT)) {
|
||||||
|
// Rounding + 1 BV -> 1 FP
|
||||||
|
if (num_parameters != 2)
|
||||||
|
m_manager->raise_exception("invalid number of parameters to to_fp");
|
||||||
|
if (!parameters[0].is_int() || !parameters[1].is_int())
|
||||||
|
m_manager->raise_exception("invalid parameter type to to_fp");
|
||||||
|
int ebits = parameters[0].get_int();
|
||||||
|
int sbits = parameters[1].get_int();
|
||||||
|
|
||||||
|
sort * fp = mk_float_sort(ebits, sbits);
|
||||||
|
symbol name("to_fp");
|
||||||
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
||||||
|
}
|
||||||
|
else if (arity == 2 &&
|
||||||
|
is_sort_of(domain[0], m_family_id, ROUNDING_MODE_SORT) &&
|
||||||
|
is_sort_of(domain[1], m_family_id, FLOAT_SORT)) {
|
||||||
|
// Rounding + 1 FP -> 1 FP
|
||||||
|
if (num_parameters != 2)
|
||||||
|
m_manager->raise_exception("invalid number of parameters to to_fp");
|
||||||
|
if (!parameters[0].is_int() || !parameters[1].is_int())
|
||||||
|
m_manager->raise_exception("invalid parameter type to to_fp");
|
||||||
|
int ebits = parameters[0].get_int();
|
||||||
|
int sbits = parameters[1].get_int();
|
||||||
|
if (!is_rm_sort(domain[0]))
|
||||||
|
m_manager->raise_exception("sort mismatch, expected first argument of RoundingMode sort");
|
||||||
|
if (!is_sort_of(domain[1], m_family_id, FLOAT_SORT))
|
||||||
|
m_manager->raise_exception("sort mismatch, expected second argument of FloatingPoint sort");
|
||||||
|
|
||||||
|
sort * fp = mk_float_sort(ebits, sbits);
|
||||||
|
symbol name("to_fp");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// .. Otherwise we only know how to convert rationals/reals.
|
// 1 Real -> 1 FP
|
||||||
if (!(num_parameters == 2 && parameters[0].is_int() && parameters[1].is_int()))
|
if (!(num_parameters == 2 && parameters[0].is_int() && parameters[1].is_int()))
|
||||||
m_manager->raise_exception("expecting two integer parameters to asFloat");
|
m_manager->raise_exception("expecting two integer parameters to to_fp");
|
||||||
if (arity != 2 && arity != 3)
|
if (arity != 2 && arity != 3)
|
||||||
m_manager->raise_exception("invalid number of arguments to asFloat operator");
|
m_manager->raise_exception("invalid number of arguments to to_fp operator");
|
||||||
if (arity == 3 && domain[2] != m_int_sort)
|
if (arity == 3 && domain[2] != m_int_sort)
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected second argument of Int sort");
|
||||||
if (!is_rm_sort(domain[0]) ||
|
if (domain[1] != m_real_sort)
|
||||||
!(domain[1] == m_real_sort || is_sort_of(domain[1], m_family_id, FLOAT_SORT)))
|
m_manager->raise_exception("sort mismatch, expected second argument of Real sort");
|
||||||
m_manager->raise_exception("sort mismatch");
|
|
||||||
|
|
||||||
sort * fp = mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
sort * fp = mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
||||||
symbol name("asFloat");
|
symbol name("to_fp");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k, num_parameters, parameters));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_to_ieee_bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * float_decl_plugin::mk_float_to_ieee_bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (!m_bv_plugin)
|
|
||||||
m_manager->raise_exception("asIEEEBV unsupported; use a logic with BV support");
|
|
||||||
if (arity != 1)
|
if (arity != 1)
|
||||||
m_manager->raise_exception("invalid number of arguments to asIEEEBV");
|
m_manager->raise_exception("invalid number of arguments to asIEEEBV");
|
||||||
if (!is_float_sort(domain[0]))
|
if (!is_float_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismatch");
|
m_manager->raise_exception("sort mismatch, expected argument of FloatingPoint sort");
|
||||||
|
|
||||||
// When the bv_decl_plugin is installed, then we know how to convert a float to an IEEE bit-vector.
|
|
||||||
unsigned float_sz = domain[0]->get_parameter(0).get_int() + domain[0]->get_parameter(1).get_int();
|
unsigned float_sz = domain[0]->get_parameter(0).get_int() + domain[0]->get_parameter(1).get_int();
|
||||||
parameter ps[] = { parameter(float_sz) };
|
parameter ps[] = { parameter(float_sz) };
|
||||||
sort * bv_srt = m_bv_plugin->mk_sort(m_bv_fid, 1, ps);
|
sort * bv_srt = m_bv_plugin->mk_sort(m_bv_fid, 1, ps);
|
||||||
|
@ -429,41 +487,34 @@ func_decl * float_decl_plugin::mk_to_ieee_bv(decl_kind k, unsigned num_parameter
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_from3bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * float_decl_plugin::mk_from3bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (!m_bv_plugin)
|
|
||||||
m_manager->raise_exception("fp unsupported; use a logic with BV support");
|
|
||||||
if (arity != 3)
|
if (arity != 3)
|
||||||
m_manager->raise_exception("invalid number of arguments to fp");
|
m_manager->raise_exception("invalid number of arguments to fp");
|
||||||
if (!is_sort_of(domain[0], m_bv_fid, BV_SORT) ||
|
if (!is_sort_of(domain[0], m_bv_fid, BV_SORT) ||
|
||||||
!is_sort_of(domain[1], m_bv_fid, BV_SORT) ||
|
!is_sort_of(domain[1], m_bv_fid, BV_SORT) ||
|
||||||
!is_sort_of(domain[2], m_bv_fid, BV_SORT))
|
!is_sort_of(domain[2], m_bv_fid, BV_SORT))
|
||||||
m_manager->raise_exception("sort mismtach");
|
m_manager->raise_exception("sort mismatch");
|
||||||
|
|
||||||
sort * fp = mk_float_sort(domain[1]->get_parameter(0).get_int(), domain[2]->get_parameter(0).get_int() + 1);
|
sort * fp = mk_float_sort(domain[1]->get_parameter(0).get_int(), domain[2]->get_parameter(0).get_int() + 1);
|
||||||
symbol name("fp");
|
symbol name("fp");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k));
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k));
|
||||||
}
|
}
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_to_fp_unsigned(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * float_decl_plugin::mk_to_ubv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
if (!m_bv_plugin)
|
if (!m_bv_plugin)
|
||||||
m_manager->raise_exception("to_fp_unsigned unsupported; use a logic with BV support");
|
m_manager->raise_exception("to_fp_unsigned unsupported; use a logic with BV support");
|
||||||
if (arity != 2)
|
if (arity != 2)
|
||||||
m_manager->raise_exception("invalid number of arguments to to_fp_unsigned");
|
m_manager->raise_exception("invalid number of arguments to to_fp_unsigned");
|
||||||
if (is_rm_sort(domain[0]))
|
if (is_rm_sort(domain[0]))
|
||||||
m_manager->raise_exception("sort mismtach");
|
m_manager->raise_exception("sort mismatch, expected first argument of RoundingMode sort");
|
||||||
if (!is_sort_of(domain[1], m_bv_fid, BV_SORT))
|
if (!is_sort_of(domain[1], m_bv_fid, BV_SORT))
|
||||||
m_manager->raise_exception("sort mismtach");
|
m_manager->raise_exception("sort mismatch, expected second argument of BV sort");
|
||||||
|
|
||||||
sort * fp = mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
sort * fp = mk_float_sort(parameters[0].get_int(), parameters[1].get_int());
|
||||||
symbol name("to_fp_unsigned");
|
symbol name("fp.t_ubv");
|
||||||
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k));
|
return m_manager->mk_func_decl(name, arity, domain, fp, func_decl_info(m_family_id, k));
|
||||||
}
|
}
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_to_ubv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
|
||||||
NOT_IMPLEMENTED_YET();
|
|
||||||
}
|
|
||||||
|
|
||||||
func_decl * float_decl_plugin::mk_to_sbv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * float_decl_plugin::mk_to_sbv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range) {
|
unsigned arity, sort * const * domain, sort * range) {
|
||||||
NOT_IMPLEMENTED_YET();
|
NOT_IMPLEMENTED_YET();
|
||||||
|
@ -498,14 +549,15 @@ func_decl * float_decl_plugin::mk_func_decl(decl_kind k, unsigned num_parameters
|
||||||
case OP_FLOAT_IS_ZERO:
|
case OP_FLOAT_IS_ZERO:
|
||||||
case OP_FLOAT_IS_NZERO:
|
case OP_FLOAT_IS_NZERO:
|
||||||
case OP_FLOAT_IS_PZERO:
|
case OP_FLOAT_IS_PZERO:
|
||||||
case OP_FLOAT_IS_SIGN_MINUS:
|
case OP_FLOAT_IS_NEGATIVE:
|
||||||
|
case OP_FLOAT_IS_POSITIVE:
|
||||||
case OP_FLOAT_IS_NAN:
|
case OP_FLOAT_IS_NAN:
|
||||||
case OP_FLOAT_IS_INF:
|
case OP_FLOAT_IS_INF:
|
||||||
case OP_FLOAT_IS_NORMAL:
|
case OP_FLOAT_IS_NORMAL:
|
||||||
case OP_FLOAT_IS_SUBNORMAL:
|
case OP_FLOAT_IS_SUBNORMAL:
|
||||||
return mk_unary_rel_decl(k, num_parameters, parameters, arity, domain, range);
|
return mk_unary_rel_decl(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_ABS:
|
case OP_FLOAT_ABS:
|
||||||
case OP_FLOAT_UMINUS:
|
case OP_FLOAT_NEG:
|
||||||
return mk_unary_decl(k, num_parameters, parameters, arity, domain, range);
|
return mk_unary_decl(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_REM:
|
case OP_FLOAT_REM:
|
||||||
case OP_FLOAT_MIN:
|
case OP_FLOAT_MIN:
|
||||||
|
@ -517,20 +569,18 @@ func_decl * float_decl_plugin::mk_func_decl(decl_kind k, unsigned num_parameters
|
||||||
return mk_rm_binary_decl(k, num_parameters, parameters, arity, domain, range);
|
return mk_rm_binary_decl(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_SUB:
|
case OP_FLOAT_SUB:
|
||||||
if (arity == 1)
|
if (arity == 1)
|
||||||
return mk_unary_decl(OP_FLOAT_UMINUS, num_parameters, parameters, arity, domain, range);
|
return mk_unary_decl(OP_FLOAT_NEG, num_parameters, parameters, arity, domain, range);
|
||||||
else
|
else
|
||||||
return mk_rm_binary_decl(k, num_parameters, parameters, arity, domain, range);
|
return mk_rm_binary_decl(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_SQRT:
|
case OP_FLOAT_SQRT:
|
||||||
case OP_FLOAT_ROUND_TO_INTEGRAL:
|
case OP_FLOAT_ROUND_TO_INTEGRAL:
|
||||||
return mk_rm_unary_decl(k, num_parameters, parameters, arity, domain, range);
|
return mk_rm_unary_decl(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_FUSED_MA:
|
case OP_FLOAT_FMA:
|
||||||
return mk_fused_ma(k, num_parameters, parameters, arity, domain, range);
|
return mk_fma(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_TO_IEEE_BV:
|
case OP_FLOAT_TO_IEEE_BV:
|
||||||
return mk_to_ieee_bv(k, num_parameters, parameters, arity, domain, range);
|
return mk_float_to_ieee_bv(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_FP:
|
case OP_FLOAT_FP:
|
||||||
return mk_from3bv(k, num_parameters, parameters, arity, domain, range);
|
return mk_from3bv(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_TO_FP_UNSIGNED:
|
|
||||||
return mk_to_fp_unsigned(k, num_parameters, parameters, arity, domain, range);
|
|
||||||
case OP_FLOAT_TO_UBV:
|
case OP_FLOAT_TO_UBV:
|
||||||
return mk_to_ubv(k, num_parameters, parameters, arity, domain, range);
|
return mk_to_ubv(k, num_parameters, parameters, arity, domain, range);
|
||||||
case OP_FLOAT_TO_SBV:
|
case OP_FLOAT_TO_SBV:
|
||||||
|
@ -544,8 +594,11 @@ func_decl * float_decl_plugin::mk_func_decl(decl_kind k, unsigned num_parameters
|
||||||
}
|
}
|
||||||
|
|
||||||
void float_decl_plugin::get_op_names(svector<builtin_name> & op_names, symbol const & logic) {
|
void float_decl_plugin::get_op_names(svector<builtin_name> & op_names, symbol const & logic) {
|
||||||
op_names.push_back(builtin_name("plusInfinity", OP_FLOAT_PLUS_INF));
|
// These are the operators from the final draft of the SMT FloatingPoint standard
|
||||||
op_names.push_back(builtin_name("minusInfinity", OP_FLOAT_MINUS_INF));
|
op_names.push_back(builtin_name("+oo", OP_FLOAT_PLUS_INF));
|
||||||
|
op_names.push_back(builtin_name("-oo", OP_FLOAT_MINUS_INF));
|
||||||
|
op_names.push_back(builtin_name("+zero", OP_FLOAT_PLUS_ZERO));
|
||||||
|
op_names.push_back(builtin_name("-zero", OP_FLOAT_MINUS_ZERO));
|
||||||
op_names.push_back(builtin_name("NaN", OP_FLOAT_NAN));
|
op_names.push_back(builtin_name("NaN", OP_FLOAT_NAN));
|
||||||
|
|
||||||
op_names.push_back(builtin_name("roundNearestTiesToEven", OP_RM_NEAREST_TIES_TO_EVEN));
|
op_names.push_back(builtin_name("roundNearestTiesToEven", OP_RM_NEAREST_TIES_TO_EVEN));
|
||||||
|
@ -554,46 +607,6 @@ void float_decl_plugin::get_op_names(svector<builtin_name> & op_names, symbol co
|
||||||
op_names.push_back(builtin_name("roundTowardNegative", OP_RM_TOWARD_NEGATIVE));
|
op_names.push_back(builtin_name("roundTowardNegative", OP_RM_TOWARD_NEGATIVE));
|
||||||
op_names.push_back(builtin_name("roundTowardZero", OP_RM_TOWARD_ZERO));
|
op_names.push_back(builtin_name("roundTowardZero", OP_RM_TOWARD_ZERO));
|
||||||
|
|
||||||
op_names.push_back(builtin_name("+", OP_FLOAT_ADD));
|
|
||||||
op_names.push_back(builtin_name("-", OP_FLOAT_SUB));
|
|
||||||
op_names.push_back(builtin_name("/", OP_FLOAT_DIV));
|
|
||||||
op_names.push_back(builtin_name("*", OP_FLOAT_MUL));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("abs", OP_FLOAT_ABS));
|
|
||||||
op_names.push_back(builtin_name("remainder", OP_FLOAT_REM));
|
|
||||||
op_names.push_back(builtin_name("fusedMA", OP_FLOAT_FUSED_MA));
|
|
||||||
op_names.push_back(builtin_name("squareRoot", OP_FLOAT_SQRT));
|
|
||||||
op_names.push_back(builtin_name("roundToIntegral", OP_FLOAT_ROUND_TO_INTEGRAL));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("==", OP_FLOAT_EQ));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("<", OP_FLOAT_LT));
|
|
||||||
op_names.push_back(builtin_name(">", OP_FLOAT_GT));
|
|
||||||
op_names.push_back(builtin_name("<=", OP_FLOAT_LE));
|
|
||||||
op_names.push_back(builtin_name(">=", OP_FLOAT_GE));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("isNaN", OP_FLOAT_IS_NAN));
|
|
||||||
op_names.push_back(builtin_name("isInfinite", OP_FLOAT_IS_INF));
|
|
||||||
op_names.push_back(builtin_name("isZero", OP_FLOAT_IS_ZERO));
|
|
||||||
op_names.push_back(builtin_name("isNZero", OP_FLOAT_IS_NZERO));
|
|
||||||
op_names.push_back(builtin_name("isPZero", OP_FLOAT_IS_PZERO));
|
|
||||||
op_names.push_back(builtin_name("isNormal", OP_FLOAT_IS_NORMAL));
|
|
||||||
op_names.push_back(builtin_name("isSubnormal", OP_FLOAT_IS_SUBNORMAL));
|
|
||||||
op_names.push_back(builtin_name("isSignMinus", OP_FLOAT_IS_SIGN_MINUS));
|
|
||||||
|
|
||||||
// Disabled min/max, clashes with user-defined min/max functions
|
|
||||||
// op_names.push_back(builtin_name("min", OP_FLOAT_MIN));
|
|
||||||
// op_names.push_back(builtin_name("max", OP_FLOAT_MAX));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("asFloat", OP_TO_FLOAT));
|
|
||||||
|
|
||||||
if (m_bv_plugin)
|
|
||||||
op_names.push_back(builtin_name("asIEEEBV", OP_TO_IEEE_BV));
|
|
||||||
|
|
||||||
// These are the operators from the final draft of the SMT FloatingPoints standard
|
|
||||||
op_names.push_back(builtin_name("+oo", OP_FLOAT_PLUS_INF));
|
|
||||||
op_names.push_back(builtin_name("-oo", OP_FLOAT_MINUS_INF));
|
|
||||||
|
|
||||||
op_names.push_back(builtin_name("RNE", OP_RM_NEAREST_TIES_TO_EVEN));
|
op_names.push_back(builtin_name("RNE", OP_RM_NEAREST_TIES_TO_EVEN));
|
||||||
op_names.push_back(builtin_name("RNA", OP_RM_NEAREST_TIES_TO_AWAY));
|
op_names.push_back(builtin_name("RNA", OP_RM_NEAREST_TIES_TO_AWAY));
|
||||||
op_names.push_back(builtin_name("RTP", OP_RM_TOWARD_POSITIVE));
|
op_names.push_back(builtin_name("RTP", OP_RM_TOWARD_POSITIVE));
|
||||||
|
@ -601,44 +614,47 @@ void float_decl_plugin::get_op_names(svector<builtin_name> & op_names, symbol co
|
||||||
op_names.push_back(builtin_name("RTZ", OP_RM_TOWARD_ZERO));
|
op_names.push_back(builtin_name("RTZ", OP_RM_TOWARD_ZERO));
|
||||||
|
|
||||||
op_names.push_back(builtin_name("fp.abs", OP_FLOAT_ABS));
|
op_names.push_back(builtin_name("fp.abs", OP_FLOAT_ABS));
|
||||||
op_names.push_back(builtin_name("fp.neg", OP_FLOAT_UMINUS));
|
op_names.push_back(builtin_name("fp.neg", OP_FLOAT_NEG));
|
||||||
op_names.push_back(builtin_name("fp.add", OP_FLOAT_ADD));
|
op_names.push_back(builtin_name("fp.add", OP_FLOAT_ADD));
|
||||||
op_names.push_back(builtin_name("fp.sub", OP_FLOAT_SUB));
|
op_names.push_back(builtin_name("fp.sub", OP_FLOAT_SUB));
|
||||||
op_names.push_back(builtin_name("fp.mul", OP_FLOAT_MUL));
|
op_names.push_back(builtin_name("fp.mul", OP_FLOAT_MUL));
|
||||||
op_names.push_back(builtin_name("fp.div", OP_FLOAT_DIV));
|
op_names.push_back(builtin_name("fp.div", OP_FLOAT_DIV));
|
||||||
op_names.push_back(builtin_name("fp.fma", OP_FLOAT_FUSED_MA));
|
op_names.push_back(builtin_name("fp.fma", OP_FLOAT_FMA));
|
||||||
op_names.push_back(builtin_name("fp.sqrt", OP_FLOAT_SQRT));
|
op_names.push_back(builtin_name("fp.sqrt", OP_FLOAT_SQRT));
|
||||||
op_names.push_back(builtin_name("fp.rem", OP_FLOAT_REM));
|
op_names.push_back(builtin_name("fp.rem", OP_FLOAT_REM));
|
||||||
op_names.push_back(builtin_name("fp.eq", OP_FLOAT_EQ));
|
op_names.push_back(builtin_name("fp.roundToIntegral", OP_FLOAT_ROUND_TO_INTEGRAL));
|
||||||
|
op_names.push_back(builtin_name("fp.min", OP_FLOAT_MIN));
|
||||||
|
op_names.push_back(builtin_name("fp.max", OP_FLOAT_MAX));
|
||||||
op_names.push_back(builtin_name("fp.leq", OP_FLOAT_LE));
|
op_names.push_back(builtin_name("fp.leq", OP_FLOAT_LE));
|
||||||
op_names.push_back(builtin_name("fp.lt", OP_FLOAT_LT));
|
op_names.push_back(builtin_name("fp.lt", OP_FLOAT_LT));
|
||||||
op_names.push_back(builtin_name("fp.geq", OP_FLOAT_GE));
|
op_names.push_back(builtin_name("fp.geq", OP_FLOAT_GE));
|
||||||
op_names.push_back(builtin_name("fp.gt", OP_FLOAT_GT));
|
op_names.push_back(builtin_name("fp.gt", OP_FLOAT_GT));
|
||||||
|
op_names.push_back(builtin_name("fp.eq", OP_FLOAT_EQ));
|
||||||
|
|
||||||
op_names.push_back(builtin_name("fp.isNormal", OP_FLOAT_IS_NORMAL));
|
op_names.push_back(builtin_name("fp.isNormal", OP_FLOAT_IS_NORMAL));
|
||||||
op_names.push_back(builtin_name("fp.isSubnormal", OP_FLOAT_IS_SUBNORMAL));
|
op_names.push_back(builtin_name("fp.isSubnormal", OP_FLOAT_IS_SUBNORMAL));
|
||||||
op_names.push_back(builtin_name("fp.isZero", OP_FLOAT_IS_ZERO));
|
op_names.push_back(builtin_name("fp.isZero", OP_FLOAT_IS_ZERO));
|
||||||
op_names.push_back(builtin_name("fp.isInfinite", OP_FLOAT_IS_INF));
|
op_names.push_back(builtin_name("fp.isInfinite", OP_FLOAT_IS_INF));
|
||||||
op_names.push_back(builtin_name("fp.isNaN", OP_FLOAT_IS_NAN));
|
op_names.push_back(builtin_name("fp.isNaN", OP_FLOAT_IS_NAN));
|
||||||
op_names.push_back(builtin_name("fp.min", OP_FLOAT_MIN));
|
op_names.push_back(builtin_name("fp.isNegative", OP_FLOAT_IS_NEGATIVE));
|
||||||
op_names.push_back(builtin_name("fp.max", OP_FLOAT_MAX));
|
op_names.push_back(builtin_name("fp.isPositive", OP_FLOAT_IS_POSITIVE));
|
||||||
op_names.push_back(builtin_name("to_fp", OP_TO_FLOAT));
|
|
||||||
|
|
||||||
if (m_bv_plugin) {
|
op_names.push_back(builtin_name("fp", OP_FLOAT_FP));
|
||||||
op_names.push_back(builtin_name("fp", OP_FLOAT_FP));
|
op_names.push_back(builtin_name("fp.to_ubv", OP_FLOAT_TO_UBV));
|
||||||
op_names.push_back(builtin_name("to_fp_unsigned", OP_FLOAT_TO_FP_UNSIGNED));
|
op_names.push_back(builtin_name("fp.to_sbv", OP_FLOAT_TO_SBV));
|
||||||
op_names.push_back(builtin_name("fp.to_ubv", OP_FLOAT_TO_UBV));
|
|
||||||
op_names.push_back(builtin_name("fp.to_sbv", OP_FLOAT_TO_SBV));
|
|
||||||
}
|
|
||||||
|
|
||||||
// op_names.push_back(builtin_name("fp.toReal", ?));
|
op_names.push_back(builtin_name("to_fp", OP_TO_FLOAT));
|
||||||
}
|
}
|
||||||
|
|
||||||
void float_decl_plugin::get_sort_names(svector<builtin_name> & sort_names, symbol const & logic) {
|
void float_decl_plugin::get_sort_names(svector<builtin_name> & sort_names, symbol const & logic) {
|
||||||
sort_names.push_back(builtin_name("FP", FLOAT_SORT));
|
sort_names.push_back(builtin_name("FloatingPoint", FLOAT_SORT));
|
||||||
sort_names.push_back(builtin_name("RoundingMode", ROUNDING_MODE_SORT));
|
sort_names.push_back(builtin_name("RoundingMode", ROUNDING_MODE_SORT));
|
||||||
|
|
||||||
// In the SMT FPA final draft, FP is called FloatingPoint
|
// The final theory supports three common FloatingPoint sorts
|
||||||
sort_names.push_back(builtin_name("FloatingPoint", FLOAT_SORT));
|
sort_names.push_back(builtin_name("Float16", FLOAT16_SORT));
|
||||||
|
sort_names.push_back(builtin_name("Float32", FLOAT32_SORT));
|
||||||
|
sort_names.push_back(builtin_name("Float64", FLOAT64_SORT));
|
||||||
|
sort_names.push_back(builtin_name("Float128", FLOAT128_SORT));
|
||||||
}
|
}
|
||||||
|
|
||||||
expr * float_decl_plugin::get_some_value(sort * s) {
|
expr * float_decl_plugin::get_some_value(sort * s) {
|
||||||
|
@ -662,6 +678,8 @@ bool float_decl_plugin::is_value(app * e) const {
|
||||||
case OP_FLOAT_VALUE:
|
case OP_FLOAT_VALUE:
|
||||||
case OP_FLOAT_PLUS_INF:
|
case OP_FLOAT_PLUS_INF:
|
||||||
case OP_FLOAT_MINUS_INF:
|
case OP_FLOAT_MINUS_INF:
|
||||||
|
case OP_FLOAT_PLUS_ZERO:
|
||||||
|
case OP_FLOAT_MINUS_ZERO:
|
||||||
case OP_FLOAT_NAN:
|
case OP_FLOAT_NAN:
|
||||||
return true;
|
return true;
|
||||||
case OP_TO_FLOAT:
|
case OP_TO_FLOAT:
|
||||||
|
|
|
@ -27,7 +27,11 @@ Revision History:
|
||||||
|
|
||||||
enum float_sort_kind {
|
enum float_sort_kind {
|
||||||
FLOAT_SORT,
|
FLOAT_SORT,
|
||||||
ROUNDING_MODE_SORT
|
ROUNDING_MODE_SORT,
|
||||||
|
FLOAT16_SORT,
|
||||||
|
FLOAT32_SORT,
|
||||||
|
FLOAT64_SORT,
|
||||||
|
FLOAT128_SORT
|
||||||
};
|
};
|
||||||
|
|
||||||
enum float_op_kind {
|
enum float_op_kind {
|
||||||
|
@ -37,22 +41,23 @@ enum float_op_kind {
|
||||||
OP_RM_TOWARD_NEGATIVE,
|
OP_RM_TOWARD_NEGATIVE,
|
||||||
OP_RM_TOWARD_ZERO,
|
OP_RM_TOWARD_ZERO,
|
||||||
|
|
||||||
|
|
||||||
OP_FLOAT_VALUE,
|
OP_FLOAT_VALUE,
|
||||||
OP_FLOAT_PLUS_INF,
|
OP_FLOAT_PLUS_INF,
|
||||||
OP_FLOAT_MINUS_INF,
|
OP_FLOAT_MINUS_INF,
|
||||||
OP_FLOAT_NAN,
|
OP_FLOAT_NAN,
|
||||||
|
OP_FLOAT_PLUS_ZERO,
|
||||||
|
OP_FLOAT_MINUS_ZERO,
|
||||||
|
|
||||||
OP_FLOAT_ADD,
|
OP_FLOAT_ADD,
|
||||||
OP_FLOAT_SUB,
|
OP_FLOAT_SUB,
|
||||||
OP_FLOAT_UMINUS,
|
OP_FLOAT_NEG,
|
||||||
OP_FLOAT_MUL,
|
OP_FLOAT_MUL,
|
||||||
OP_FLOAT_DIV,
|
OP_FLOAT_DIV,
|
||||||
OP_FLOAT_REM,
|
OP_FLOAT_REM,
|
||||||
OP_FLOAT_ABS,
|
OP_FLOAT_ABS,
|
||||||
OP_FLOAT_MIN,
|
OP_FLOAT_MIN,
|
||||||
OP_FLOAT_MAX,
|
OP_FLOAT_MAX,
|
||||||
OP_FLOAT_FUSED_MA, // x*y + z
|
OP_FLOAT_FMA, // x*y + z
|
||||||
OP_FLOAT_SQRT,
|
OP_FLOAT_SQRT,
|
||||||
OP_FLOAT_ROUND_TO_INTEGRAL,
|
OP_FLOAT_ROUND_TO_INTEGRAL,
|
||||||
|
|
||||||
|
@ -68,13 +73,14 @@ enum float_op_kind {
|
||||||
OP_FLOAT_IS_SUBNORMAL,
|
OP_FLOAT_IS_SUBNORMAL,
|
||||||
OP_FLOAT_IS_PZERO,
|
OP_FLOAT_IS_PZERO,
|
||||||
OP_FLOAT_IS_NZERO,
|
OP_FLOAT_IS_NZERO,
|
||||||
OP_FLOAT_IS_SIGN_MINUS,
|
OP_FLOAT_IS_NEGATIVE,
|
||||||
|
OP_FLOAT_IS_POSITIVE,
|
||||||
|
|
||||||
OP_TO_FLOAT,
|
OP_TO_FLOAT,
|
||||||
OP_TO_IEEE_BV,
|
OP_FLOAT_TO_IEEE_BV,
|
||||||
|
|
||||||
OP_FLOAT_FP,
|
OP_FLOAT_FP,
|
||||||
OP_FLOAT_TO_FP_UNSIGNED,
|
OP_FLOAT_TO_FP,
|
||||||
OP_FLOAT_TO_UBV,
|
OP_FLOAT_TO_UBV,
|
||||||
OP_FLOAT_TO_SBV,
|
OP_FLOAT_TO_SBV,
|
||||||
OP_FLOAT_TO_REAL,
|
OP_FLOAT_TO_REAL,
|
||||||
|
@ -125,16 +131,14 @@ class float_decl_plugin : public decl_plugin {
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_rm_unary_decl(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_rm_unary_decl(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_fused_ma(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_fma(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_to_float(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_to_float(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_to_ieee_bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_float_to_ieee_bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_from3bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_from3bv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_to_fp_unsigned(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
|
||||||
func_decl * mk_to_ubv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_to_ubv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
unsigned arity, sort * const * domain, sort * range);
|
unsigned arity, sort * const * domain, sort * range);
|
||||||
func_decl * mk_to_sbv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
func_decl * mk_to_sbv(decl_kind k, unsigned num_parameters, parameter const * parameters,
|
||||||
|
@ -195,6 +199,7 @@ public:
|
||||||
family_id get_fid() const { return m_fid; }
|
family_id get_fid() const { return m_fid; }
|
||||||
family_id get_family_id() const { return m_fid; }
|
family_id get_family_id() const { return m_fid; }
|
||||||
arith_util & au() { return m_a_util; }
|
arith_util & au() { return m_a_util; }
|
||||||
|
float_decl_plugin & plugin() { return *m_plugin; }
|
||||||
|
|
||||||
sort * mk_float_sort(unsigned ebits, unsigned sbits);
|
sort * mk_float_sort(unsigned ebits, unsigned sbits);
|
||||||
sort * mk_rm_sort() { return m().mk_sort(m_fid, ROUNDING_MODE_SORT); }
|
sort * mk_rm_sort() { return m().mk_sort(m_fid, ROUNDING_MODE_SORT); }
|
||||||
|
@ -242,16 +247,16 @@ public:
|
||||||
app * mk_mul(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_MUL, arg1, arg2, arg3); }
|
app * mk_mul(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_MUL, arg1, arg2, arg3); }
|
||||||
app * mk_sub(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_SUB, arg1, arg2, arg3); }
|
app * mk_sub(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_SUB, arg1, arg2, arg3); }
|
||||||
app * mk_div(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_DIV, arg1, arg2, arg3); }
|
app * mk_div(expr * arg1, expr * arg2, expr * arg3) { return m().mk_app(m_fid, OP_FLOAT_DIV, arg1, arg2, arg3); }
|
||||||
app * mk_uminus(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_UMINUS, arg1); }
|
app * mk_neg(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_NEG, arg1); }
|
||||||
app * mk_rem(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_REM, arg1, arg2); }
|
app * mk_rem(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_REM, arg1, arg2); }
|
||||||
app * mk_max(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_MAX, arg1, arg2); }
|
app * mk_max(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_MAX, arg1, arg2); }
|
||||||
app * mk_min(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_MIN, arg1, arg2); }
|
app * mk_min(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_MIN, arg1, arg2); }
|
||||||
app * mk_abs(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_ABS, arg1); }
|
app * mk_abs(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_ABS, arg1); }
|
||||||
app * mk_sqrt(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_SQRT, arg1, arg2); }
|
app * mk_sqrt(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_SQRT, arg1, arg2); }
|
||||||
app * mk_round(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_ROUND_TO_INTEGRAL, arg1, arg2); }
|
app * mk_round(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_ROUND_TO_INTEGRAL, arg1, arg2); }
|
||||||
app * mk_fused_ma(expr * arg1, expr * arg2, expr * arg3, expr * arg4) {
|
app * mk_fma(expr * arg1, expr * arg2, expr * arg3, expr * arg4) {
|
||||||
expr * args[4] = { arg1, arg2, arg3, arg4 };
|
expr * args[4] = { arg1, arg2, arg3, arg4 };
|
||||||
return m().mk_app(m_fid, OP_FLOAT_FUSED_MA, 4, args);
|
return m().mk_app(m_fid, OP_FLOAT_FMA, 4, args);
|
||||||
}
|
}
|
||||||
|
|
||||||
app * mk_float_eq(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_EQ, arg1, arg2); }
|
app * mk_float_eq(expr * arg1, expr * arg2) { return m().mk_app(m_fid, OP_FLOAT_EQ, arg1, arg2); }
|
||||||
|
@ -267,11 +272,13 @@ public:
|
||||||
app * mk_is_subnormal(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_SUBNORMAL, arg1); }
|
app * mk_is_subnormal(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_SUBNORMAL, arg1); }
|
||||||
app * mk_is_nzero(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_NZERO, arg1); }
|
app * mk_is_nzero(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_NZERO, arg1); }
|
||||||
app * mk_is_pzero(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_PZERO, arg1); }
|
app * mk_is_pzero(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_PZERO, arg1); }
|
||||||
app * mk_is_sign_minus(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_SIGN_MINUS, arg1); }
|
app * mk_is_sign_minus(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_NEGATIVE, arg1); }
|
||||||
|
app * mk_is_positive(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_POSITIVE, arg1); }
|
||||||
|
app * mk_is_negative(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_IS_NEGATIVE, arg1); }
|
||||||
|
|
||||||
bool is_uminus(expr * a) { return is_app_of(a, m_fid, OP_FLOAT_UMINUS); }
|
bool is_neg(expr * a) { return is_app_of(a, m_fid, OP_FLOAT_NEG); }
|
||||||
|
|
||||||
app * mk_to_ieee_bv(expr * arg1) { return m().mk_app(m_fid, OP_TO_IEEE_BV, arg1); }
|
app * mk_float_to_ieee_bv(expr * arg1) { return m().mk_app(m_fid, OP_FLOAT_TO_IEEE_BV, arg1); }
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
|
@ -136,12 +136,13 @@ void fpa2bv_converter::mk_const(func_decl * f, expr_ref & result) {
|
||||||
unsigned sbits = m_util.get_sbits(srt);
|
unsigned sbits = m_util.get_sbits(srt);
|
||||||
|
|
||||||
expr_ref sgn(m), s(m), e(m);
|
expr_ref sgn(m), s(m), e(m);
|
||||||
sort_ref s_sgn(m), s_sig(m), s_exp(m);
|
|
||||||
s_sgn = m_bv_util.mk_sort(1);
|
|
||||||
s_sig = m_bv_util.mk_sort(sbits-1);
|
|
||||||
s_exp = m_bv_util.mk_sort(ebits);
|
|
||||||
|
|
||||||
#ifdef Z3DEBUG
|
#ifdef Z3DEBUG
|
||||||
|
sort_ref s_sgn(m), s_sig(m), s_exp(m);
|
||||||
|
s_sgn = m_bv_util.mk_sort(1);
|
||||||
|
s_sig = m_bv_util.mk_sort(sbits - 1);
|
||||||
|
s_exp = m_bv_util.mk_sort(ebits);
|
||||||
|
|
||||||
std::string p("fpa2bv");
|
std::string p("fpa2bv");
|
||||||
std::string name = f->get_name().str();
|
std::string name = f->get_name().str();
|
||||||
|
|
||||||
|
@ -149,9 +150,17 @@ void fpa2bv_converter::mk_const(func_decl * f, expr_ref & result) {
|
||||||
s = m.mk_fresh_const((p + "_sig_" + name).c_str(), s_sig);
|
s = m.mk_fresh_const((p + "_sig_" + name).c_str(), s_sig);
|
||||||
e = m.mk_fresh_const((p + "_exp_" + name).c_str(), s_exp);
|
e = m.mk_fresh_const((p + "_exp_" + name).c_str(), s_exp);
|
||||||
#else
|
#else
|
||||||
sgn = m.mk_fresh_const(0, s_sgn);
|
expr_ref bv(m);
|
||||||
s = m.mk_fresh_const(0, s_sig);
|
unsigned bv_sz = 1 + ebits + (sbits - 1);
|
||||||
e = m.mk_fresh_const(0, s_exp);
|
bv = m.mk_fresh_const(0, m_bv_util.mk_sort(bv_sz));
|
||||||
|
|
||||||
|
sgn = m_bv_util.mk_extract(bv_sz - 1, bv_sz - 1, bv);
|
||||||
|
e = m_bv_util.mk_extract(bv_sz - 2, sbits - 1, bv);
|
||||||
|
s = m_bv_util.mk_extract(sbits - 2, 0, bv);
|
||||||
|
|
||||||
|
SASSERT(m_bv_util.get_bv_size(sgn) == 1);
|
||||||
|
SASSERT(m_bv_util.get_bv_size(s) == sbits-1);
|
||||||
|
SASSERT(m_bv_util.get_bv_size(e) == ebits);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
mk_triple(sgn, s, e, result);
|
mk_triple(sgn, s, e, result);
|
||||||
|
@ -595,12 +604,12 @@ void fpa2bv_converter::mk_add(func_decl * f, unsigned num, expr * const * args,
|
||||||
void fpa2bv_converter::mk_sub(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_sub(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 3);
|
SASSERT(num == 3);
|
||||||
expr_ref t(m);
|
expr_ref t(m);
|
||||||
mk_uminus(f, 1, &args[2], t);
|
mk_neg(f, 1, &args[2], t);
|
||||||
expr * nargs[3] = { args[0], args[1], t };
|
expr * nargs[3] = { args[0], args[1], t };
|
||||||
mk_add(f, 3, nargs, result);
|
mk_add(f, 3, nargs, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_uminus(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_neg(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 1);
|
SASSERT(num == 1);
|
||||||
expr * sgn, * s, * e;
|
expr * sgn, * s, * e;
|
||||||
split(args[0], sgn, s, e);
|
split(args[0], sgn, s, e);
|
||||||
|
@ -906,7 +915,7 @@ void fpa2bv_converter::mk_div(func_decl * f, unsigned num, expr * const * args,
|
||||||
TRACE("fpa2bv_div", tout << "DIV = " << mk_ismt2_pp(result, m) << std::endl; );
|
TRACE("fpa2bv_div", tout << "DIV = " << mk_ismt2_pp(result, m) << std::endl; );
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_remainder(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_rem(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 2);
|
SASSERT(num == 2);
|
||||||
|
|
||||||
// Remainder is always exact, so there is no rounding mode.
|
// Remainder is always exact, so there is no rounding mode.
|
||||||
|
@ -1114,7 +1123,7 @@ void fpa2bv_converter::mk_max(func_decl * f, unsigned num, expr * const * args,
|
||||||
mk_triple(r_sgn, r_sig, r_exp, result);
|
mk_triple(r_sgn, r_sig, r_exp, result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_fusedma(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_fma(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 4);
|
SASSERT(num == 4);
|
||||||
|
|
||||||
// fusedma means (x * y) + z
|
// fusedma means (x * y) + z
|
||||||
|
@ -1835,9 +1844,18 @@ void fpa2bv_converter::mk_is_subnormal(func_decl * f, unsigned num, expr * const
|
||||||
mk_is_denormal(args[0], result);
|
mk_is_denormal(args[0], result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_is_sign_minus(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_is_negative(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 1);
|
SASSERT(num == 1);
|
||||||
mk_is_neg(args[0], result);
|
mk_is_neg(args[0], result);
|
||||||
|
TRACE("fpa2bv_is_negative", tout << "result = " << mk_ismt2_pp(result, m) << std::endl;);
|
||||||
|
}
|
||||||
|
|
||||||
|
void fpa2bv_converter::mk_is_positive(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
|
SASSERT(num == 1);
|
||||||
|
expr_ref t1(m), t2(m);
|
||||||
|
mk_is_nan(args[0], t1);
|
||||||
|
mk_is_pos(args[0], t2);
|
||||||
|
result = m.mk_and(m.mk_not(t1), t2);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_to_float(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_to_float(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
|
@ -2123,7 +2141,7 @@ void fpa2bv_converter::mk_to_float(func_decl * f, unsigned num, expr * const * a
|
||||||
void fpa2bv_converter::mk_to_ieee_bv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_to_ieee_bv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(num == 1);
|
SASSERT(num == 1);
|
||||||
expr * sgn, * s, * e;
|
expr * sgn, * s, * e;
|
||||||
split(args[0], sgn, s, e);
|
split(args[0], sgn, s, e);
|
||||||
result = m_bv_util.mk_concat(m_bv_util.mk_concat(sgn, e), s);
|
result = m_bv_util.mk_concat(m_bv_util.mk_concat(sgn, e), s);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -2132,19 +2150,55 @@ void fpa2bv_converter::mk_fp(func_decl * f, unsigned num, expr * const * args, e
|
||||||
mk_triple(args[0], args[2], args[1], result);
|
mk_triple(args[0], args[2], args[1], result);
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_to_fp_unsigned(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
|
||||||
NOT_IMPLEMENTED_YET();
|
|
||||||
}
|
|
||||||
|
|
||||||
void fpa2bv_converter::mk_to_ubv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_to_ubv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
|
SASSERT(num == 2);
|
||||||
|
SASSERT(f->get_num_parameters() == 1);
|
||||||
|
SASSERT(f->get_parameter(0).is_int());
|
||||||
|
|
||||||
|
unsigned ebits = m_util.get_ebits(f->get_range());
|
||||||
|
unsigned sbits = m_util.get_sbits(f->get_range());
|
||||||
|
int width = f->get_parameter(0).get_int();
|
||||||
|
|
||||||
|
expr * rm = args[0];
|
||||||
|
expr * x = args[1];
|
||||||
|
|
||||||
|
expr * sgn, *s, *e;
|
||||||
|
split(x, sgn, s, e);
|
||||||
|
|
||||||
NOT_IMPLEMENTED_YET();
|
NOT_IMPLEMENTED_YET();
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_to_sbv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_to_sbv(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
|
SASSERT(num == 2);
|
||||||
|
SASSERT(f->get_num_parameters() == 1);
|
||||||
|
SASSERT(f->get_parameter(0).is_int());
|
||||||
|
|
||||||
|
unsigned ebits = m_util.get_ebits(f->get_range());
|
||||||
|
unsigned sbits = m_util.get_sbits(f->get_range());
|
||||||
|
int width = f->get_parameter(0).get_int();
|
||||||
|
|
||||||
|
expr * rm = args[0];
|
||||||
|
expr * x = args[1];
|
||||||
|
|
||||||
|
expr * sgn, *s, *e;
|
||||||
|
split(x, sgn, s, e);
|
||||||
|
|
||||||
NOT_IMPLEMENTED_YET();
|
NOT_IMPLEMENTED_YET();
|
||||||
}
|
}
|
||||||
|
|
||||||
void fpa2bv_converter::mk_to_real(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
void fpa2bv_converter::mk_to_real(func_decl * f, unsigned num, expr * const * args, expr_ref & result) {
|
||||||
|
SASSERT(num == 1);
|
||||||
|
|
||||||
|
unsigned ebits = m_util.get_ebits(f->get_range());
|
||||||
|
unsigned sbits = m_util.get_sbits(f->get_range());
|
||||||
|
int width = f->get_parameter(0).get_int();
|
||||||
|
|
||||||
|
expr * rm = args[0];
|
||||||
|
expr * x = args[1];
|
||||||
|
|
||||||
|
expr * sgn, *s, *e;
|
||||||
|
split(x, sgn, s, e);
|
||||||
|
|
||||||
NOT_IMPLEMENTED_YET();
|
NOT_IMPLEMENTED_YET();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -94,14 +94,14 @@ public:
|
||||||
|
|
||||||
void mk_add(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_add(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_sub(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_sub(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_uminus(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_neg(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_mul(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_mul(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_div(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_div(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_remainder(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_rem(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_abs(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_abs(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_min(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_min(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_max(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_max(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_fusedma(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_fma(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_sqrt(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_sqrt(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_round_to_integral(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_round_to_integral(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
|
|
||||||
|
@ -114,7 +114,8 @@ public:
|
||||||
void mk_is_zero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_zero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_nzero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_nzero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_pzero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_pzero(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_sign_minus(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_negative(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
|
void mk_is_positive(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_nan(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_nan(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_inf(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_inf(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
void mk_is_normal(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
void mk_is_normal(func_decl * f, unsigned num, expr * const * args, expr_ref & result);
|
||||||
|
|
|
@ -65,8 +65,6 @@ struct fpa2bv_rewriter_cfg : public default_rewriter_cfg {
|
||||||
|
|
||||||
bool max_steps_exceeded(unsigned num_steps) const {
|
bool max_steps_exceeded(unsigned num_steps) const {
|
||||||
cooperate("fpa2bv");
|
cooperate("fpa2bv");
|
||||||
if (memory::get_allocation_size() > m_max_memory)
|
|
||||||
throw tactic_exception(TACTIC_MAX_MEMORY_MSG);
|
|
||||||
return num_steps > m_max_steps;
|
return num_steps > m_max_steps;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -117,17 +115,19 @@ struct fpa2bv_rewriter_cfg : public default_rewriter_cfg {
|
||||||
case OP_FLOAT_VALUE: m_conv.mk_value(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_VALUE: m_conv.mk_value(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_PLUS_INF: m_conv.mk_plus_inf(f, result); return BR_DONE;
|
case OP_FLOAT_PLUS_INF: m_conv.mk_plus_inf(f, result); return BR_DONE;
|
||||||
case OP_FLOAT_MINUS_INF: m_conv.mk_minus_inf(f, result); return BR_DONE;
|
case OP_FLOAT_MINUS_INF: m_conv.mk_minus_inf(f, result); return BR_DONE;
|
||||||
|
case OP_FLOAT_PLUS_ZERO: m_conv.mk_pzero(f, result); return BR_DONE;
|
||||||
|
case OP_FLOAT_MINUS_ZERO: m_conv.mk_nzero(f, result); return BR_DONE;
|
||||||
case OP_FLOAT_NAN: m_conv.mk_nan(f, result); return BR_DONE;
|
case OP_FLOAT_NAN: m_conv.mk_nan(f, result); return BR_DONE;
|
||||||
case OP_FLOAT_ADD: m_conv.mk_add(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_ADD: m_conv.mk_add(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_SUB: m_conv.mk_sub(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_SUB: m_conv.mk_sub(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_UMINUS: m_conv.mk_uminus(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_NEG: m_conv.mk_neg(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_MUL: m_conv.mk_mul(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_MUL: m_conv.mk_mul(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_DIV: m_conv.mk_div(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_DIV: m_conv.mk_div(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_REM: m_conv.mk_remainder(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_REM: m_conv.mk_rem(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_ABS: m_conv.mk_abs(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_ABS: m_conv.mk_abs(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_MIN: m_conv.mk_min(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_MIN: m_conv.mk_min(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_MAX: m_conv.mk_max(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_MAX: m_conv.mk_max(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_FUSED_MA: m_conv.mk_fusedma(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_FMA: m_conv.mk_fma(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_SQRT: m_conv.mk_sqrt(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_SQRT: m_conv.mk_sqrt(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_ROUND_TO_INTEGRAL: m_conv.mk_round_to_integral(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_ROUND_TO_INTEGRAL: m_conv.mk_round_to_integral(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_EQ: m_conv.mk_float_eq(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_EQ: m_conv.mk_float_eq(f, num, args, result); return BR_DONE;
|
||||||
|
@ -142,18 +142,18 @@ struct fpa2bv_rewriter_cfg : public default_rewriter_cfg {
|
||||||
case OP_FLOAT_IS_INF: m_conv.mk_is_inf(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_IS_INF: m_conv.mk_is_inf(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_IS_NORMAL: m_conv.mk_is_normal(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_IS_NORMAL: m_conv.mk_is_normal(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_IS_SUBNORMAL: m_conv.mk_is_subnormal(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_IS_SUBNORMAL: m_conv.mk_is_subnormal(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_IS_SIGN_MINUS: m_conv.mk_is_sign_minus(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_IS_POSITIVE: m_conv.mk_is_positive(f, num, args, result); return BR_DONE;
|
||||||
|
case OP_FLOAT_IS_NEGATIVE: m_conv.mk_is_negative(f, num, args, result); return BR_DONE;
|
||||||
case OP_TO_FLOAT: m_conv.mk_to_float(f, num, args, result); return BR_DONE;
|
case OP_TO_FLOAT: m_conv.mk_to_float(f, num, args, result); return BR_DONE;
|
||||||
case OP_TO_IEEE_BV: m_conv.mk_to_ieee_bv(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_TO_IEEE_BV: m_conv.mk_to_ieee_bv(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_FP: m_conv.mk_fp(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_FP: m_conv.mk_fp(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_TO_FP_UNSIGNED: m_conv.mk_to_fp_unsigned(f, num, args, result); return BR_DONE;
|
|
||||||
case OP_FLOAT_TO_UBV: m_conv.mk_to_ubv(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_TO_UBV: m_conv.mk_to_ubv(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_TO_SBV: m_conv.mk_to_sbv(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_TO_SBV: m_conv.mk_to_sbv(f, num, args, result); return BR_DONE;
|
||||||
case OP_FLOAT_TO_REAL: m_conv.mk_to_real(f, num, args, result); return BR_DONE;
|
case OP_FLOAT_TO_REAL: m_conv.mk_to_real(f, num, args, result); return BR_DONE;
|
||||||
default:
|
default:
|
||||||
TRACE("fpa2bv", tout << "unsupported operator: " << f->get_name() << "\n";
|
TRACE("fpa2bv", tout << "unsupported operator: " << f->get_name() << "\n";
|
||||||
for (unsigned i = 0; i < num; i++) tout << mk_ismt2_pp(args[i], m()) << std::endl;);
|
for (unsigned i = 0; i < num; i++) tout << mk_ismt2_pp(args[i], m()) << std::endl;);
|
||||||
throw tactic_exception("NYI");
|
NOT_IMPLEMENTED_YET();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -36,17 +36,17 @@ br_status float_rewriter::mk_app_core(func_decl * f, unsigned num_args, expr * c
|
||||||
br_status st = BR_FAILED;
|
br_status st = BR_FAILED;
|
||||||
SASSERT(f->get_family_id() == get_fid());
|
SASSERT(f->get_family_id() == get_fid());
|
||||||
switch (f->get_decl_kind()) {
|
switch (f->get_decl_kind()) {
|
||||||
case OP_TO_FLOAT: st = mk_to_float(f, num_args, args, result); break;
|
case OP_TO_FLOAT: st = mk_to_fp(f, num_args, args, result); break;
|
||||||
case OP_FLOAT_ADD: SASSERT(num_args == 3); st = mk_add(args[0], args[1], args[2], result); break;
|
case OP_FLOAT_ADD: SASSERT(num_args == 3); st = mk_add(args[0], args[1], args[2], result); break;
|
||||||
case OP_FLOAT_SUB: SASSERT(num_args == 3); st = mk_sub(args[0], args[1], args[2], result); break;
|
case OP_FLOAT_SUB: SASSERT(num_args == 3); st = mk_sub(args[0], args[1], args[2], result); break;
|
||||||
case OP_FLOAT_UMINUS: SASSERT(num_args == 1); st = mk_uminus(args[0], result); break;
|
case OP_FLOAT_NEG: SASSERT(num_args == 1); st = mk_neg(args[0], result); break;
|
||||||
case OP_FLOAT_MUL: SASSERT(num_args == 3); st = mk_mul(args[0], args[1], args[2], result); break;
|
case OP_FLOAT_MUL: SASSERT(num_args == 3); st = mk_mul(args[0], args[1], args[2], result); break;
|
||||||
case OP_FLOAT_DIV: SASSERT(num_args == 3); st = mk_div(args[0], args[1], args[2], result); break;
|
case OP_FLOAT_DIV: SASSERT(num_args == 3); st = mk_div(args[0], args[1], args[2], result); break;
|
||||||
case OP_FLOAT_REM: SASSERT(num_args == 2); st = mk_rem(args[0], args[1], result); break;
|
case OP_FLOAT_REM: SASSERT(num_args == 2); st = mk_rem(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_ABS: SASSERT(num_args == 1); st = mk_abs(args[0], result); break;
|
case OP_FLOAT_ABS: SASSERT(num_args == 1); st = mk_abs(args[0], result); break;
|
||||||
case OP_FLOAT_MIN: SASSERT(num_args == 2); st = mk_min(args[0], args[1], result); break;
|
case OP_FLOAT_MIN: SASSERT(num_args == 2); st = mk_min(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_MAX: SASSERT(num_args == 2); st = mk_max(args[0], args[1], result); break;
|
case OP_FLOAT_MAX: SASSERT(num_args == 2); st = mk_max(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_FUSED_MA: SASSERT(num_args == 4); st = mk_fused_ma(args[0], args[1], args[2], args[3], result); break;
|
case OP_FLOAT_FMA: SASSERT(num_args == 4); st = mk_fma(args[0], args[1], args[2], args[3], result); break;
|
||||||
case OP_FLOAT_SQRT: SASSERT(num_args == 2); st = mk_sqrt(args[0], args[1], result); break;
|
case OP_FLOAT_SQRT: SASSERT(num_args == 2); st = mk_sqrt(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_ROUND_TO_INTEGRAL: SASSERT(num_args == 2); st = mk_round(args[0], args[1], result); break;
|
case OP_FLOAT_ROUND_TO_INTEGRAL: SASSERT(num_args == 2); st = mk_round(args[0], args[1], result); break;
|
||||||
|
|
||||||
|
@ -62,10 +62,10 @@ br_status float_rewriter::mk_app_core(func_decl * f, unsigned num_args, expr * c
|
||||||
case OP_FLOAT_IS_INF: SASSERT(num_args == 1); st = mk_is_inf(args[0], result); break;
|
case OP_FLOAT_IS_INF: SASSERT(num_args == 1); st = mk_is_inf(args[0], result); break;
|
||||||
case OP_FLOAT_IS_NORMAL: SASSERT(num_args == 1); st = mk_is_normal(args[0], result); break;
|
case OP_FLOAT_IS_NORMAL: SASSERT(num_args == 1); st = mk_is_normal(args[0], result); break;
|
||||||
case OP_FLOAT_IS_SUBNORMAL: SASSERT(num_args == 1); st = mk_is_subnormal(args[0], result); break;
|
case OP_FLOAT_IS_SUBNORMAL: SASSERT(num_args == 1); st = mk_is_subnormal(args[0], result); break;
|
||||||
case OP_FLOAT_IS_SIGN_MINUS: SASSERT(num_args == 1); st = mk_is_sign_minus(args[0], result); break;
|
case OP_FLOAT_IS_NEGATIVE: SASSERT(num_args == 1); st = mk_is_negative(args[0], result); break;
|
||||||
case OP_TO_IEEE_BV: SASSERT(num_args == 1); st = mk_to_ieee_bv(args[0], result); break;
|
case OP_FLOAT_IS_POSITIVE: SASSERT(num_args == 1); st = mk_is_positive(args[0], result); break;
|
||||||
|
case OP_FLOAT_TO_IEEE_BV: SASSERT(num_args == 1); st = mk_to_ieee_bv(args[0], result); break;
|
||||||
case OP_FLOAT_FP: SASSERT(num_args == 3); st = mk_fp(args[0], args[1], args[2], result); break;
|
case OP_FLOAT_FP: SASSERT(num_args == 3); st = mk_fp(args[0], args[1], args[2], result); break;
|
||||||
case OP_FLOAT_TO_FP_UNSIGNED: SASSERT(num_args == 2); st = mk_to_fp_unsigned(args[0], args[1], result); break;
|
|
||||||
case OP_FLOAT_TO_UBV: SASSERT(num_args == 2); st = mk_to_ubv(args[0], args[1], result); break;
|
case OP_FLOAT_TO_UBV: SASSERT(num_args == 2); st = mk_to_ubv(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_TO_SBV: SASSERT(num_args == 2); st = mk_to_sbv(args[0], args[1], result); break;
|
case OP_FLOAT_TO_SBV: SASSERT(num_args == 2); st = mk_to_sbv(args[0], args[1], result); break;
|
||||||
case OP_FLOAT_TO_REAL: SASSERT(num_args == 1); st = mk_to_real(args[0], result); break;
|
case OP_FLOAT_TO_REAL: SASSERT(num_args == 1); st = mk_to_real(args[0], result); break;
|
||||||
|
@ -73,7 +73,7 @@ br_status float_rewriter::mk_app_core(func_decl * f, unsigned num_args, expr * c
|
||||||
return st;
|
return st;
|
||||||
}
|
}
|
||||||
|
|
||||||
br_status float_rewriter::mk_to_float(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result) {
|
br_status float_rewriter::mk_to_fp(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result) {
|
||||||
SASSERT(f->get_num_parameters() == 2);
|
SASSERT(f->get_num_parameters() == 2);
|
||||||
SASSERT(f->get_parameter(0).is_int());
|
SASSERT(f->get_parameter(0).is_int());
|
||||||
SASSERT(f->get_parameter(1).is_int());
|
SASSERT(f->get_parameter(1).is_int());
|
||||||
|
@ -154,7 +154,7 @@ br_status float_rewriter::mk_add(expr * arg1, expr * arg2, expr * arg3, expr_ref
|
||||||
|
|
||||||
br_status float_rewriter::mk_sub(expr * arg1, expr * arg2, expr * arg3, expr_ref & result) {
|
br_status float_rewriter::mk_sub(expr * arg1, expr * arg2, expr * arg3, expr_ref & result) {
|
||||||
// a - b = a + (-b)
|
// a - b = a + (-b)
|
||||||
result = m_util.mk_add(arg1, arg2, m_util.mk_uminus(arg3));
|
result = m_util.mk_add(arg1, arg2, m_util.mk_neg(arg3));
|
||||||
return BR_REWRITE2;
|
return BR_REWRITE2;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -188,7 +188,7 @@ br_status float_rewriter::mk_div(expr * arg1, expr * arg2, expr * arg3, expr_ref
|
||||||
return BR_FAILED;
|
return BR_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
br_status float_rewriter::mk_uminus(expr * arg1, expr_ref & result) {
|
br_status float_rewriter::mk_neg(expr * arg1, expr_ref & result) {
|
||||||
if (m_util.is_nan(arg1)) {
|
if (m_util.is_nan(arg1)) {
|
||||||
// -nan --> nan
|
// -nan --> nan
|
||||||
result = arg1;
|
result = arg1;
|
||||||
|
@ -204,7 +204,7 @@ br_status float_rewriter::mk_uminus(expr * arg1, expr_ref & result) {
|
||||||
result = m_util.mk_plus_inf(m().get_sort(arg1));
|
result = m_util.mk_plus_inf(m().get_sort(arg1));
|
||||||
return BR_DONE;
|
return BR_DONE;
|
||||||
}
|
}
|
||||||
if (m_util.is_uminus(arg1)) {
|
if (m_util.is_neg(arg1)) {
|
||||||
// - - a --> a
|
// - - a --> a
|
||||||
result = to_app(arg1)->get_arg(0);
|
result = to_app(arg1)->get_arg(0);
|
||||||
return BR_DONE;
|
return BR_DONE;
|
||||||
|
@ -239,7 +239,7 @@ br_status float_rewriter::mk_abs(expr * arg1, expr_ref & result) {
|
||||||
return BR_DONE;
|
return BR_DONE;
|
||||||
}
|
}
|
||||||
result = m().mk_ite(m_util.mk_is_sign_minus(arg1),
|
result = m().mk_ite(m_util.mk_is_sign_minus(arg1),
|
||||||
m_util.mk_uminus(arg1),
|
m_util.mk_neg(arg1),
|
||||||
arg1);
|
arg1);
|
||||||
return BR_REWRITE2;
|
return BR_REWRITE2;
|
||||||
}
|
}
|
||||||
|
@ -284,7 +284,7 @@ br_status float_rewriter::mk_max(expr * arg1, expr * arg2, expr_ref & result) {
|
||||||
return BR_REWRITE_FULL;
|
return BR_REWRITE_FULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
br_status float_rewriter::mk_fused_ma(expr * arg1, expr * arg2, expr * arg3, expr * arg4, expr_ref & result) {
|
br_status float_rewriter::mk_fma(expr * arg1, expr * arg2, expr * arg3, expr * arg4, expr_ref & result) {
|
||||||
mpf_rounding_mode rm;
|
mpf_rounding_mode rm;
|
||||||
if (m_util.is_rm_value(arg1, rm)) {
|
if (m_util.is_rm_value(arg1, rm)) {
|
||||||
scoped_mpf v2(m_util.fm()), v3(m_util.fm()), v4(m_util.fm());
|
scoped_mpf v2(m_util.fm()), v3(m_util.fm()), v4(m_util.fm());
|
||||||
|
@ -480,7 +480,7 @@ br_status float_rewriter::mk_is_subnormal(expr * arg1, expr_ref & result) {
|
||||||
return BR_FAILED;
|
return BR_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
br_status float_rewriter::mk_is_sign_minus(expr * arg1, expr_ref & result) {
|
br_status float_rewriter::mk_is_negative(expr * arg1, expr_ref & result) {
|
||||||
scoped_mpf v(m_util.fm());
|
scoped_mpf v(m_util.fm());
|
||||||
if (m_util.is_value(arg1, v)) {
|
if (m_util.is_value(arg1, v)) {
|
||||||
result = (m_util.fm().is_neg(v)) ? m().mk_true() : m().mk_false();
|
result = (m_util.fm().is_neg(v)) ? m().mk_true() : m().mk_false();
|
||||||
|
@ -490,6 +490,17 @@ br_status float_rewriter::mk_is_sign_minus(expr * arg1, expr_ref & result) {
|
||||||
return BR_FAILED;
|
return BR_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
br_status float_rewriter::mk_is_positive(expr * arg1, expr_ref & result) {
|
||||||
|
scoped_mpf v(m_util.fm());
|
||||||
|
if (m_util.is_value(arg1, v)) {
|
||||||
|
result = (m_util.fm().is_neg(v) || m_util.fm().is_nan(v)) ? m().mk_false() : m().mk_true();
|
||||||
|
return BR_DONE;
|
||||||
|
}
|
||||||
|
|
||||||
|
return BR_FAILED;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// This the SMT =
|
// This the SMT =
|
||||||
br_status float_rewriter::mk_eq_core(expr * arg1, expr * arg2, expr_ref & result) {
|
br_status float_rewriter::mk_eq_core(expr * arg1, expr * arg2, expr_ref & result) {
|
||||||
scoped_mpf v1(m_util.fm()), v2(m_util.fm());
|
scoped_mpf v1(m_util.fm()), v2(m_util.fm());
|
||||||
|
@ -532,10 +543,6 @@ br_status float_rewriter::mk_fp(expr * arg1, expr * arg2, expr * arg3, expr_ref
|
||||||
return BR_FAILED;
|
return BR_FAILED;
|
||||||
}
|
}
|
||||||
|
|
||||||
br_status float_rewriter::mk_to_fp_unsigned(expr * arg1, expr * arg2, expr_ref & result) {
|
|
||||||
return BR_FAILED;
|
|
||||||
}
|
|
||||||
|
|
||||||
br_status float_rewriter::mk_to_ubv(expr * arg1, expr * arg2, expr_ref & result) {
|
br_status float_rewriter::mk_to_ubv(expr * arg1, expr * arg2, expr_ref & result) {
|
||||||
return BR_FAILED;
|
return BR_FAILED;
|
||||||
}
|
}
|
||||||
|
|
|
@ -45,17 +45,16 @@ public:
|
||||||
br_status mk_app_core(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result);
|
br_status mk_app_core(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result);
|
||||||
br_status mk_eq_core(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_eq_core(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
|
|
||||||
br_status mk_to_float(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result);
|
|
||||||
br_status mk_add(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
br_status mk_add(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
||||||
br_status mk_sub(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
br_status mk_sub(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
||||||
br_status mk_mul(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
br_status mk_mul(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
||||||
br_status mk_div(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
br_status mk_div(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
||||||
br_status mk_uminus(expr * arg1, expr_ref & result);
|
br_status mk_neg(expr * arg1, expr_ref & result);
|
||||||
br_status mk_rem(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_rem(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_abs(expr * arg1, expr_ref & result);
|
br_status mk_abs(expr * arg1, expr_ref & result);
|
||||||
br_status mk_min(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_min(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_max(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_max(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_fused_ma(expr * arg1, expr * arg2, expr * arg3, expr * arg4, expr_ref & result);
|
br_status mk_fma(expr * arg1, expr * arg2, expr * arg3, expr * arg4, expr_ref & result);
|
||||||
br_status mk_sqrt(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_sqrt(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_round(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_round(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_float_eq(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_float_eq(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
|
@ -70,10 +69,12 @@ public:
|
||||||
br_status mk_is_inf(expr * arg1, expr_ref & result);
|
br_status mk_is_inf(expr * arg1, expr_ref & result);
|
||||||
br_status mk_is_normal(expr * arg1, expr_ref & result);
|
br_status mk_is_normal(expr * arg1, expr_ref & result);
|
||||||
br_status mk_is_subnormal(expr * arg1, expr_ref & result);
|
br_status mk_is_subnormal(expr * arg1, expr_ref & result);
|
||||||
br_status mk_is_sign_minus(expr * arg1, expr_ref & result);
|
br_status mk_is_negative(expr * arg1, expr_ref & result);
|
||||||
|
br_status mk_is_positive(expr * arg1, expr_ref & result);
|
||||||
|
|
||||||
br_status mk_to_ieee_bv(expr * arg1, expr_ref & result);
|
br_status mk_to_ieee_bv(expr * arg1, expr_ref & result);
|
||||||
|
|
||||||
|
br_status mk_to_fp(func_decl * f, unsigned num_args, expr * const * args, expr_ref & result);
|
||||||
br_status mk_fp(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
br_status mk_fp(expr * arg1, expr * arg2, expr * arg3, expr_ref & result);
|
||||||
br_status mk_to_fp_unsigned(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_to_fp_unsigned(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
br_status mk_to_ubv(expr * arg1, expr * arg2, expr_ref & result);
|
br_status mk_to_ubv(expr * arg1, expr * arg2, expr_ref & result);
|
||||||
|
|
|
@ -241,7 +241,6 @@ protected:
|
||||||
symbol m_produce_models;
|
symbol m_produce_models;
|
||||||
symbol m_produce_assignments;
|
symbol m_produce_assignments;
|
||||||
symbol m_produce_interpolants;
|
symbol m_produce_interpolants;
|
||||||
symbol m_check_interpolants;
|
|
||||||
symbol m_regular_output_channel;
|
symbol m_regular_output_channel;
|
||||||
symbol m_diagnostic_output_channel;
|
symbol m_diagnostic_output_channel;
|
||||||
symbol m_random_seed;
|
symbol m_random_seed;
|
||||||
|
@ -256,7 +255,6 @@ protected:
|
||||||
s == m_print_success || s == m_print_warning || s == m_expand_definitions ||
|
s == m_print_success || s == m_print_warning || s == m_expand_definitions ||
|
||||||
s == m_interactive_mode || s == m_produce_proofs || s == m_produce_unsat_cores ||
|
s == m_interactive_mode || s == m_produce_proofs || s == m_produce_unsat_cores ||
|
||||||
s == m_produce_models || s == m_produce_assignments || s == m_produce_interpolants ||
|
s == m_produce_models || s == m_produce_assignments || s == m_produce_interpolants ||
|
||||||
s == m_check_interpolants ||
|
|
||||||
s == m_regular_output_channel || s == m_diagnostic_output_channel ||
|
s == m_regular_output_channel || s == m_diagnostic_output_channel ||
|
||||||
s == m_random_seed || s == m_verbosity || s == m_global_decls;
|
s == m_random_seed || s == m_verbosity || s == m_global_decls;
|
||||||
}
|
}
|
||||||
|
@ -275,7 +273,6 @@ public:
|
||||||
m_produce_models(":produce-models"),
|
m_produce_models(":produce-models"),
|
||||||
m_produce_assignments(":produce-assignments"),
|
m_produce_assignments(":produce-assignments"),
|
||||||
m_produce_interpolants(":produce-interpolants"),
|
m_produce_interpolants(":produce-interpolants"),
|
||||||
m_check_interpolants(":check-interpolants"),
|
|
||||||
m_regular_output_channel(":regular-output-channel"),
|
m_regular_output_channel(":regular-output-channel"),
|
||||||
m_diagnostic_output_channel(":diagnostic-output-channel"),
|
m_diagnostic_output_channel(":diagnostic-output-channel"),
|
||||||
m_random_seed(":random-seed"),
|
m_random_seed(":random-seed"),
|
||||||
|
@ -347,9 +344,6 @@ class set_option_cmd : public set_get_option_cmd {
|
||||||
check_not_initialized(ctx, m_produce_interpolants);
|
check_not_initialized(ctx, m_produce_interpolants);
|
||||||
ctx.set_produce_interpolants(to_bool(value));
|
ctx.set_produce_interpolants(to_bool(value));
|
||||||
}
|
}
|
||||||
else if (m_option == m_check_interpolants) {
|
|
||||||
ctx.set_check_interpolants(to_bool(value));
|
|
||||||
}
|
|
||||||
else if (m_option == m_produce_unsat_cores) {
|
else if (m_option == m_produce_unsat_cores) {
|
||||||
check_not_initialized(ctx, m_produce_unsat_cores);
|
check_not_initialized(ctx, m_produce_unsat_cores);
|
||||||
ctx.set_produce_unsat_cores(to_bool(value));
|
ctx.set_produce_unsat_cores(to_bool(value));
|
||||||
|
|
|
@ -410,10 +410,6 @@ void cmd_context::set_produce_interpolants(bool f) {
|
||||||
// set_solver_factory(mk_smt_solver_factory());
|
// set_solver_factory(mk_smt_solver_factory());
|
||||||
}
|
}
|
||||||
|
|
||||||
void cmd_context::set_check_interpolants(bool f) {
|
|
||||||
m_params.m_check_interpolants = f;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool cmd_context::produce_models() const {
|
bool cmd_context::produce_models() const {
|
||||||
return m_params.m_model;
|
return m_params.m_model;
|
||||||
}
|
}
|
||||||
|
@ -427,10 +423,6 @@ bool cmd_context::produce_interpolants() const {
|
||||||
return m_params.m_proof;
|
return m_params.m_proof;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool cmd_context::check_interpolants() const {
|
|
||||||
return m_params.m_check_interpolants;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool cmd_context::produce_unsat_cores() const {
|
bool cmd_context::produce_unsat_cores() const {
|
||||||
return m_params.m_unsat_core;
|
return m_params.m_unsat_core;
|
||||||
}
|
}
|
||||||
|
|
|
@ -296,7 +296,6 @@ public:
|
||||||
bool produce_models() const;
|
bool produce_models() const;
|
||||||
bool produce_proofs() const;
|
bool produce_proofs() const;
|
||||||
bool produce_interpolants() const;
|
bool produce_interpolants() const;
|
||||||
bool check_interpolants() const;
|
|
||||||
bool produce_unsat_cores() const;
|
bool produce_unsat_cores() const;
|
||||||
bool well_sorted_check_enabled() const;
|
bool well_sorted_check_enabled() const;
|
||||||
bool validate_model_enabled() const;
|
bool validate_model_enabled() const;
|
||||||
|
@ -304,7 +303,6 @@ public:
|
||||||
void set_produce_unsat_cores(bool flag);
|
void set_produce_unsat_cores(bool flag);
|
||||||
void set_produce_proofs(bool flag);
|
void set_produce_proofs(bool flag);
|
||||||
void set_produce_interpolants(bool flag);
|
void set_produce_interpolants(bool flag);
|
||||||
void set_check_interpolants(bool flag);
|
|
||||||
bool produce_assignments() const { return m_produce_assignments; }
|
bool produce_assignments() const { return m_produce_assignments; }
|
||||||
void set_produce_assignments(bool flag) { m_produce_assignments = flag; }
|
void set_produce_assignments(bool flag) { m_produce_assignments = flag; }
|
||||||
void set_status(status st) { m_status = st; }
|
void set_status(status st) { m_status = st; }
|
||||||
|
|
|
@ -61,9 +61,6 @@ void context_params::set(char const * param, char const * value) {
|
||||||
else if (p == "proof") {
|
else if (p == "proof") {
|
||||||
set_bool(m_proof, param, value);
|
set_bool(m_proof, param, value);
|
||||||
}
|
}
|
||||||
else if (p == "check_interpolants") {
|
|
||||||
set_bool(m_check_interpolants, param, value);
|
|
||||||
}
|
|
||||||
else if (p == "model") {
|
else if (p == "model") {
|
||||||
set_bool(m_model, param, value);
|
set_bool(m_model, param, value);
|
||||||
}
|
}
|
||||||
|
@ -105,7 +102,6 @@ void context_params::updt_params(params_ref const & p) {
|
||||||
m_well_sorted_check = p.get_bool("type_check", p.get_bool("well_sorted_check", true));
|
m_well_sorted_check = p.get_bool("type_check", p.get_bool("well_sorted_check", true));
|
||||||
m_auto_config = p.get_bool("auto_config", true);
|
m_auto_config = p.get_bool("auto_config", true);
|
||||||
m_proof = p.get_bool("proof", false);
|
m_proof = p.get_bool("proof", false);
|
||||||
m_check_interpolants = p.get_bool("check_interpolants", false);
|
|
||||||
m_model = p.get_bool("model", true);
|
m_model = p.get_bool("model", true);
|
||||||
m_model_validate = p.get_bool("model_validate", false);
|
m_model_validate = p.get_bool("model_validate", false);
|
||||||
m_trace = p.get_bool("trace", false);
|
m_trace = p.get_bool("trace", false);
|
||||||
|
@ -120,7 +116,6 @@ void context_params::collect_param_descrs(param_descrs & d) {
|
||||||
d.insert("well_sorted_check", CPK_BOOL, "type checker", "true");
|
d.insert("well_sorted_check", CPK_BOOL, "type checker", "true");
|
||||||
d.insert("type_check", CPK_BOOL, "type checker (alias for well_sorted_check)", "true");
|
d.insert("type_check", CPK_BOOL, "type checker (alias for well_sorted_check)", "true");
|
||||||
d.insert("auto_config", CPK_BOOL, "use heuristics to automatically select solver and configure it", "true");
|
d.insert("auto_config", CPK_BOOL, "use heuristics to automatically select solver and configure it", "true");
|
||||||
d.insert("check_interpolants", CPK_BOOL, "check correctness of interpolants", "false");
|
|
||||||
d.insert("model_validate", CPK_BOOL, "validate models produced by solvers", "false");
|
d.insert("model_validate", CPK_BOOL, "validate models produced by solvers", "false");
|
||||||
d.insert("trace", CPK_BOOL, "trace generation for VCC", "false");
|
d.insert("trace", CPK_BOOL, "trace generation for VCC", "false");
|
||||||
d.insert("trace_file_name", CPK_STRING, "trace out file name (see option 'trace')", "z3.log");
|
d.insert("trace_file_name", CPK_STRING, "trace out file name (see option 'trace')", "z3.log");
|
||||||
|
|
|
@ -30,7 +30,6 @@ public:
|
||||||
bool m_auto_config;
|
bool m_auto_config;
|
||||||
bool m_proof;
|
bool m_proof;
|
||||||
bool m_interpolants;
|
bool m_interpolants;
|
||||||
bool m_check_interpolants;
|
|
||||||
bool m_debug_ref_count;
|
bool m_debug_ref_count;
|
||||||
bool m_trace;
|
bool m_trace;
|
||||||
std::string m_trace_file_name;
|
std::string m_trace_file_name;
|
||||||
|
|
|
@ -65,7 +65,7 @@ static void show_interpolant_and_maybe_check(cmd_context & ctx,
|
||||||
s.cleanup();
|
s.cleanup();
|
||||||
|
|
||||||
// verify, for the paranoid...
|
// verify, for the paranoid...
|
||||||
if(check || ctx.check_interpolants()){
|
if(check || interp_params(m_params).check()){
|
||||||
std::ostringstream err;
|
std::ostringstream err;
|
||||||
ast_manager &_m = ctx.m();
|
ast_manager &_m = ctx.m();
|
||||||
|
|
||||||
|
|
|
@ -104,6 +104,8 @@ namespace Duality {
|
||||||
|
|
||||||
FuncDecl RenumberPred(const FuncDecl &f, int n);
|
FuncDecl RenumberPred(const FuncDecl &f, int n);
|
||||||
|
|
||||||
|
FuncDecl NumberPred(const FuncDecl &f, int n);
|
||||||
|
|
||||||
Term ExtractStores(hash_map<ast, Term> &memo, const Term &t, std::vector<expr> &cnstrs, hash_map<ast,expr> &renaming);
|
Term ExtractStores(hash_map<ast, Term> &memo, const Term &t, std::vector<expr> &cnstrs, hash_map<ast,expr> &renaming);
|
||||||
|
|
||||||
|
|
||||||
|
@ -488,9 +490,10 @@ protected:
|
||||||
std::vector<Edge *> Incoming;
|
std::vector<Edge *> Incoming;
|
||||||
Term dual;
|
Term dual;
|
||||||
Node *map;
|
Node *map;
|
||||||
|
unsigned recursion_bound;
|
||||||
|
|
||||||
Node(const FuncDecl &_Name, const Transformer &_Annotation, const Transformer &_Bound, const Transformer &_Underapprox, const Term &_dual, RPFP *_owner, int _number)
|
Node(const FuncDecl &_Name, const Transformer &_Annotation, const Transformer &_Bound, const Transformer &_Underapprox, const Term &_dual, RPFP *_owner, int _number)
|
||||||
: Name(_Name), Annotation(_Annotation), Bound(_Bound), Underapprox(_Underapprox), dual(_dual) {owner = _owner; number = _number; Outgoing = 0;}
|
: Name(_Name), Annotation(_Annotation), Bound(_Bound), Underapprox(_Underapprox), dual(_dual) {owner = _owner; number = _number; Outgoing = 0; recursion_bound = UINT_MAX;}
|
||||||
};
|
};
|
||||||
|
|
||||||
/** Create a node in the graph. The input is a term R(v_1...v_n)
|
/** Create a node in the graph. The input is a term R(v_1...v_n)
|
||||||
|
@ -829,7 +832,7 @@ protected:
|
||||||
#ifdef _WINDOWS
|
#ifdef _WINDOWS
|
||||||
__declspec(dllexport)
|
__declspec(dllexport)
|
||||||
#endif
|
#endif
|
||||||
void FromClauses(const std::vector<Term> &clauses);
|
void FromClauses(const std::vector<Term> &clauses, const std::vector<unsigned> *bounds = 0);
|
||||||
|
|
||||||
void FromFixpointContext(fixedpoint fp, std::vector<Term> &queries);
|
void FromFixpointContext(fixedpoint fp, std::vector<Term> &queries);
|
||||||
|
|
||||||
|
@ -902,6 +905,10 @@ protected:
|
||||||
|
|
||||||
int CumulativeDecisions();
|
int CumulativeDecisions();
|
||||||
|
|
||||||
|
void GreedyReduceNodes(std::vector<Node *> &nodes);
|
||||||
|
|
||||||
|
check_result CheckWithConstrainedNodes(std::vector<Node *> &posnodes,std::vector<Node *> &negnodes);
|
||||||
|
|
||||||
solver &slvr(){
|
solver &slvr(){
|
||||||
return *ls->slvr;
|
return *ls->slvr;
|
||||||
}
|
}
|
||||||
|
|
|
@ -125,8 +125,18 @@ namespace Duality {
|
||||||
|
|
||||||
void timer_stop(const char *name){
|
void timer_stop(const char *name){
|
||||||
if(current->name != name || !current->parent){
|
if(current->name != name || !current->parent){
|
||||||
|
#if 0
|
||||||
std::cerr << "imbalanced timer_start and timer_stop";
|
std::cerr << "imbalanced timer_start and timer_stop";
|
||||||
exit(1);
|
exit(1);
|
||||||
|
#endif
|
||||||
|
// in case we lost a timer stop due to an exception
|
||||||
|
while(current->name != name && current->parent)
|
||||||
|
current = current->parent;
|
||||||
|
if(current->parent){
|
||||||
|
current->time += (current_time() - current->start_time);
|
||||||
|
current = current->parent;
|
||||||
|
}
|
||||||
|
return;
|
||||||
}
|
}
|
||||||
current->time += (current_time() - current->start_time);
|
current->time += (current_time() - current->start_time);
|
||||||
current = current->parent;
|
current = current->parent;
|
||||||
|
|
|
@ -2816,6 +2816,62 @@ namespace Duality {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void foobar(){
|
||||||
|
}
|
||||||
|
|
||||||
|
void RPFP::GreedyReduceNodes(std::vector<Node *> &nodes){
|
||||||
|
std::vector<expr> lits;
|
||||||
|
for(unsigned i = 0; i < nodes.size(); i++){
|
||||||
|
Term b; std::vector<Term> v;
|
||||||
|
RedVars(nodes[i], b, v);
|
||||||
|
lits.push_back(!b);
|
||||||
|
expr bv = dualModel.eval(b);
|
||||||
|
if(eq(bv,ctx.bool_val(true))){
|
||||||
|
check_result res = slvr_check(lits.size(),&lits[0]);
|
||||||
|
if(res == unsat)
|
||||||
|
lits.pop_back();
|
||||||
|
else
|
||||||
|
foobar();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
check_result RPFP::CheckWithConstrainedNodes(std::vector<Node *> &posnodes,std::vector<Node *> &negnodes){
|
||||||
|
timer_start("Check");
|
||||||
|
std::vector<expr> lits;
|
||||||
|
for(unsigned i = 0; i < posnodes.size(); i++){
|
||||||
|
Term b; std::vector<Term> v;
|
||||||
|
RedVars(posnodes[i], b, v);
|
||||||
|
lits.push_back(b);
|
||||||
|
}
|
||||||
|
for(unsigned i = 0; i < negnodes.size(); i++){
|
||||||
|
Term b; std::vector<Term> v;
|
||||||
|
RedVars(negnodes[i], b, v);
|
||||||
|
lits.push_back(!b);
|
||||||
|
}
|
||||||
|
check_result res = slvr_check(lits.size(),&lits[0]);
|
||||||
|
if(res == unsat && posnodes.size()){
|
||||||
|
lits.resize(posnodes.size());
|
||||||
|
res = slvr_check(lits.size(),&lits[0]);
|
||||||
|
}
|
||||||
|
dualModel = slvr().get_model();
|
||||||
|
#if 0
|
||||||
|
if(!dualModel.null()){
|
||||||
|
std::cout << "posnodes called:\n";
|
||||||
|
for(unsigned i = 0; i < posnodes.size(); i++)
|
||||||
|
if(!Empty(posnodes[i]))
|
||||||
|
std::cout << posnodes[i]->Name.name() << "\n";
|
||||||
|
std::cout << "negnodes called:\n";
|
||||||
|
for(unsigned i = 0; i < negnodes.size(); i++)
|
||||||
|
if(!Empty(negnodes[i]))
|
||||||
|
std::cout << negnodes[i]->Name.name() << "\n";
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
timer_stop("Check");
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void RPFP_caching::FilterCore(std::vector<expr> &core, std::vector<expr> &full_core){
|
void RPFP_caching::FilterCore(std::vector<expr> &core, std::vector<expr> &full_core){
|
||||||
hash_set<ast> core_set;
|
hash_set<ast> core_set;
|
||||||
std::copy(full_core.begin(),full_core.end(),std::inserter(core_set,core_set.begin()));
|
std::copy(full_core.begin(),full_core.end(),std::inserter(core_set,core_set.begin()));
|
||||||
|
@ -3333,6 +3389,17 @@ namespace Duality {
|
||||||
return ctx.function(name.c_str(), arity, &domain[0], f.range());
|
return ctx.function(name.c_str(), arity, &domain[0], f.range());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Z3User::FuncDecl Z3User::NumberPred(const FuncDecl &f, int n)
|
||||||
|
{
|
||||||
|
std::string name = f.name().str();
|
||||||
|
name = name + "_" + string_of_int(n);
|
||||||
|
int arity = f.arity();
|
||||||
|
std::vector<sort> domain;
|
||||||
|
for(int i = 0; i < arity; i++)
|
||||||
|
domain.push_back(f.domain(i));
|
||||||
|
return ctx.function(name.c_str(), arity, &domain[0], f.range());
|
||||||
|
}
|
||||||
|
|
||||||
// Scan the clause body for occurrences of the predicate unknowns
|
// Scan the clause body for occurrences of the predicate unknowns
|
||||||
|
|
||||||
RPFP::Term RPFP::ScanBody(hash_map<ast,Term> &memo,
|
RPFP::Term RPFP::ScanBody(hash_map<ast,Term> &memo,
|
||||||
|
@ -3570,7 +3637,7 @@ namespace Duality {
|
||||||
|
|
||||||
#define USE_QE_LITE
|
#define USE_QE_LITE
|
||||||
|
|
||||||
void RPFP::FromClauses(const std::vector<Term> &unskolemized_clauses){
|
void RPFP::FromClauses(const std::vector<Term> &unskolemized_clauses, const std::vector<unsigned> *bounds){
|
||||||
hash_map<func_decl,Node *> pmap;
|
hash_map<func_decl,Node *> pmap;
|
||||||
func_decl fail_pred = ctx.fresh_func_decl("@Fail", ctx.bool_sort());
|
func_decl fail_pred = ctx.fresh_func_decl("@Fail", ctx.bool_sort());
|
||||||
|
|
||||||
|
@ -3663,6 +3730,7 @@ namespace Duality {
|
||||||
pmap[R] = node;
|
pmap[R] = node;
|
||||||
if (is_query)
|
if (is_query)
|
||||||
node->Bound = CreateRelation(std::vector<Term>(), ctx.bool_val(false));
|
node->Bound = CreateRelation(std::vector<Term>(), ctx.bool_val(false));
|
||||||
|
node->recursion_bound = bounds ? 0 : UINT_MAX;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -3728,6 +3796,8 @@ namespace Duality {
|
||||||
Transformer T = CreateTransformer(Relparams,Indparams,body);
|
Transformer T = CreateTransformer(Relparams,Indparams,body);
|
||||||
Edge *edge = CreateEdge(Parent,T,Children);
|
Edge *edge = CreateEdge(Parent,T,Children);
|
||||||
edge->labeled = labeled;; // remember for label extraction
|
edge->labeled = labeled;; // remember for label extraction
|
||||||
|
if(bounds)
|
||||||
|
Parent->recursion_bound = std::max(Parent->recursion_bound,(*bounds)[i]);
|
||||||
// edges.push_back(edge);
|
// edges.push_back(edge);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -33,6 +33,7 @@ Revision History:
|
||||||
#include <map>
|
#include <map>
|
||||||
#include <list>
|
#include <list>
|
||||||
#include <iterator>
|
#include <iterator>
|
||||||
|
#include <sstream>
|
||||||
|
|
||||||
// TODO: make these official options or get rid of them
|
// TODO: make these official options or get rid of them
|
||||||
|
|
||||||
|
@ -254,6 +255,13 @@ namespace Duality {
|
||||||
|
|
||||||
/** Called when done expanding a tree */
|
/** Called when done expanding a tree */
|
||||||
virtual void Done() {}
|
virtual void Done() {}
|
||||||
|
|
||||||
|
/** Ask whether a node should be used/unused in the tree. Returns,
|
||||||
|
1 if yes, -1 if no, and 0 if don't care. */
|
||||||
|
|
||||||
|
virtual int UseNode(Node *node){
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
/** The Proposer class proposes conjectures eagerly. These can come
|
/** The Proposer class proposes conjectures eagerly. These can come
|
||||||
|
@ -301,10 +309,11 @@ namespace Duality {
|
||||||
hash_set<Node *> overapproxes;
|
hash_set<Node *> overapproxes;
|
||||||
std::vector<Proposer *> proposers;
|
std::vector<Proposer *> proposers;
|
||||||
std::string ConjectureFile;
|
std::string ConjectureFile;
|
||||||
|
bool stratified_inlining_done;
|
||||||
|
|
||||||
#ifdef BOUNDED
|
#ifdef BOUNDED
|
||||||
struct Counter {
|
struct Counter {
|
||||||
int val;
|
unsigned val;
|
||||||
Counter(){val = 0;}
|
Counter(){val = 0;}
|
||||||
};
|
};
|
||||||
typedef std::map<Node *,Counter> NodeToCounter;
|
typedef std::map<Node *,Counter> NodeToCounter;
|
||||||
|
@ -313,6 +322,13 @@ namespace Duality {
|
||||||
|
|
||||||
/** Solve the problem. */
|
/** Solve the problem. */
|
||||||
virtual bool Solve(){
|
virtual bool Solve(){
|
||||||
|
PreSolve();
|
||||||
|
bool res = SolveMain(); // does the actual work
|
||||||
|
PostSolve();
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
void PreSolve(){
|
||||||
reporter = Report ? CreateStdoutReporter(rpfp) : new Reporter(rpfp);
|
reporter = Report ? CreateStdoutReporter(rpfp) : new Reporter(rpfp);
|
||||||
conj_reporter = ConjectureFile.empty() ? 0 : CreateConjectureFileReporter(rpfp,ConjectureFile);
|
conj_reporter = ConjectureFile.empty() ? 0 : CreateConjectureFileReporter(rpfp,ConjectureFile);
|
||||||
#ifndef LOCALIZE_CONJECTURES
|
#ifndef LOCALIZE_CONJECTURES
|
||||||
|
@ -321,6 +337,10 @@ namespace Duality {
|
||||||
heuristic = !cex.get_tree() ? (Heuristic *)(new LocalHeuristic(rpfp))
|
heuristic = !cex.get_tree() ? (Heuristic *)(new LocalHeuristic(rpfp))
|
||||||
: (Heuristic *)(new ReplayHeuristic(rpfp,cex));
|
: (Heuristic *)(new ReplayHeuristic(rpfp,cex));
|
||||||
#endif
|
#endif
|
||||||
|
// determine if we are recursion bounded
|
||||||
|
for(unsigned i = 0; i < rpfp->nodes.size(); i++)
|
||||||
|
if(rpfp->nodes[i]->recursion_bound < UINT_MAX)
|
||||||
|
RecursionBound = 0;
|
||||||
cex.clear(); // in case we didn't use it for heuristic
|
cex.clear(); // in case we didn't use it for heuristic
|
||||||
if(unwinding) delete unwinding;
|
if(unwinding) delete unwinding;
|
||||||
unwinding = new RPFP(rpfp->ls);
|
unwinding = new RPFP(rpfp->ls);
|
||||||
|
@ -337,9 +357,10 @@ namespace Duality {
|
||||||
return false;
|
return false;
|
||||||
#endif
|
#endif
|
||||||
StratifiedLeafCount = -1;
|
StratifiedLeafCount = -1;
|
||||||
timer_start("SolveMain");
|
stratified_inlining_done = false;
|
||||||
bool res = SolveMain(); // does the actual work
|
}
|
||||||
timer_stop("SolveMain");
|
|
||||||
|
void PostSolve(){
|
||||||
// print_profile(std::cout);
|
// print_profile(std::cout);
|
||||||
delete indset;
|
delete indset;
|
||||||
delete heuristic;
|
delete heuristic;
|
||||||
|
@ -349,7 +370,16 @@ namespace Duality {
|
||||||
delete conj_reporter;
|
delete conj_reporter;
|
||||||
for(unsigned i = 0; i < proposers.size(); i++)
|
for(unsigned i = 0; i < proposers.size(); i++)
|
||||||
delete proposers[i];
|
delete proposers[i];
|
||||||
return res;
|
}
|
||||||
|
|
||||||
|
bool RecheckBounds(){
|
||||||
|
for(unsigned i = 0; i < unwinding->nodes.size(); i++){
|
||||||
|
Node *node = unwinding->nodes[i];
|
||||||
|
node->Bound = node->map->Bound;
|
||||||
|
if(!SatisfyUpperBound(node))
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
void CreateInitialUnwinding(){
|
void CreateInitialUnwinding(){
|
||||||
|
@ -412,6 +442,7 @@ namespace Duality {
|
||||||
bool StratifiedInlining; // Do stratified inlining as preprocessing step
|
bool StratifiedInlining; // Do stratified inlining as preprocessing step
|
||||||
int RecursionBound; // Recursion bound for bounded verification
|
int RecursionBound; // Recursion bound for bounded verification
|
||||||
bool BatchExpand;
|
bool BatchExpand;
|
||||||
|
bool EnableRestarts;
|
||||||
|
|
||||||
bool SetBoolOption(bool &opt, const std::string &value){
|
bool SetBoolOption(bool &opt, const std::string &value){
|
||||||
if(value == "0") {
|
if(value == "0") {
|
||||||
|
@ -459,9 +490,12 @@ namespace Duality {
|
||||||
if(option == "conjecture_file"){
|
if(option == "conjecture_file"){
|
||||||
ConjectureFile = value;
|
ConjectureFile = value;
|
||||||
}
|
}
|
||||||
|
if(option == "enable_restarts"){
|
||||||
|
return SetBoolOption(EnableRestarts,value);
|
||||||
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/** Create an instance of a node in the unwinding. Set its
|
/** Create an instance of a node in the unwinding. Set its
|
||||||
annotation to true, and mark it unexpanded. */
|
annotation to true, and mark it unexpanded. */
|
||||||
Node* CreateNodeInstance(Node *node, int number = 0){
|
Node* CreateNodeInstance(Node *node, int number = 0){
|
||||||
|
@ -768,6 +802,15 @@ namespace Duality {
|
||||||
annot.Simplify();
|
annot.Simplify();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool NodeSolutionFromIndSetFull(Node *node){
|
||||||
|
std::vector<Node *> &insts = insts_of_node[node];
|
||||||
|
for(unsigned j = 0; j < insts.size(); j++)
|
||||||
|
if(indset->Contains(insts[j]))
|
||||||
|
if(insts[j]->Annotation.IsFull())
|
||||||
|
return true;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
bool recursionBounded;
|
bool recursionBounded;
|
||||||
|
|
||||||
/** See if the solution might be bounded. */
|
/** See if the solution might be bounded. */
|
||||||
|
@ -780,10 +823,8 @@ namespace Duality {
|
||||||
std::vector<Node *> &insts = insts_of_node[node];
|
std::vector<Node *> &insts = insts_of_node[node];
|
||||||
for(unsigned j = 0; j < insts.size(); j++)
|
for(unsigned j = 0; j < insts.size(); j++)
|
||||||
if(indset->Contains(insts[j]))
|
if(indset->Contains(insts[j]))
|
||||||
if(NodePastRecursionBound(insts[j])){
|
if(NodePastRecursionBound(insts[j],true))
|
||||||
recursionBounded = true;
|
recursionBounded = true;
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -801,12 +842,18 @@ namespace Duality {
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef BOUNDED
|
#ifdef BOUNDED
|
||||||
bool NodePastRecursionBound(Node *node){
|
bool NodePastRecursionBound(Node *node, bool report = false){
|
||||||
if(RecursionBound < 0) return false;
|
if(RecursionBound < 0) return false;
|
||||||
NodeToCounter &backs = back_edges[node];
|
NodeToCounter &backs = back_edges[node];
|
||||||
for(NodeToCounter::iterator it = backs.begin(), en = backs.end(); it != en; ++it){
|
for(NodeToCounter::iterator it = backs.begin(), en = backs.end(); it != en; ++it){
|
||||||
if(it->second.val > RecursionBound)
|
if(it->second.val > it->first->recursion_bound){
|
||||||
|
if(report){
|
||||||
|
std::ostringstream os;
|
||||||
|
os << "cut off " << it->first->Name.name() << " at depth " << it->second.val;
|
||||||
|
reporter->Message(os.str());
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
@ -892,6 +939,9 @@ namespace Duality {
|
||||||
*/
|
*/
|
||||||
|
|
||||||
bool DoStratifiedInlining(){
|
bool DoStratifiedInlining(){
|
||||||
|
if(stratified_inlining_done)
|
||||||
|
return true;
|
||||||
|
stratified_inlining_done = true;
|
||||||
DoTopoSort();
|
DoTopoSort();
|
||||||
int depth = 1; // TODO: make this an option
|
int depth = 1; // TODO: make this an option
|
||||||
std::vector<hash_map<Node *,Node *> > unfolding_levels(depth+1);
|
std::vector<hash_map<Node *,Node *> > unfolding_levels(depth+1);
|
||||||
|
@ -1034,10 +1084,17 @@ namespace Duality {
|
||||||
h->SetOldNode(node);
|
h->SetOldNode(node);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool SolveMain(){
|
||||||
|
timer_start("SolveMain");
|
||||||
|
bool res = SolveMainInt(); // does the actual work
|
||||||
|
timer_stop("SolveMain");
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
/** This does the actual solving work. We try to generate
|
/** This does the actual solving work. We try to generate
|
||||||
candidates for extension. If we succed, we extend the
|
candidates for extension. If we succed, we extend the
|
||||||
unwinding. If we fail, we have a solution. */
|
unwinding. If we fail, we have a solution. */
|
||||||
bool SolveMain(){
|
bool SolveMainInt(){
|
||||||
if(StratifiedInlining && !DoStratifiedInlining())
|
if(StratifiedInlining && !DoStratifiedInlining())
|
||||||
return false;
|
return false;
|
||||||
#ifdef BOUNDED
|
#ifdef BOUNDED
|
||||||
|
@ -1405,16 +1462,18 @@ namespace Duality {
|
||||||
slvr.pop(1);
|
slvr.pop(1);
|
||||||
delete checker;
|
delete checker;
|
||||||
#else
|
#else
|
||||||
RPFP_caching::scoped_solver_for_edge ssfe(gen_cands_rpfp,edge,true /* models */, true /*axioms*/);
|
if(!NodeSolutionFromIndSetFull(edge->Parent)){
|
||||||
gen_cands_rpfp->Push();
|
RPFP_caching::scoped_solver_for_edge ssfe(gen_cands_rpfp,edge,true /* models */, true /*axioms*/);
|
||||||
Node *root = CheckerForEdgeClone(edge,gen_cands_rpfp);
|
gen_cands_rpfp->Push();
|
||||||
if(gen_cands_rpfp->Check(root) != unsat){
|
Node *root = CheckerForEdgeClone(edge,gen_cands_rpfp);
|
||||||
Candidate candidate;
|
if(gen_cands_rpfp->Check(root) != unsat){
|
||||||
ExtractCandidateFromCex(edge,gen_cands_rpfp,root,candidate);
|
Candidate candidate;
|
||||||
reporter->InductionFailure(edge,candidate.Children);
|
ExtractCandidateFromCex(edge,gen_cands_rpfp,root,candidate);
|
||||||
candidates.push_back(candidate);
|
reporter->InductionFailure(edge,candidate.Children);
|
||||||
|
candidates.push_back(candidate);
|
||||||
|
}
|
||||||
|
gen_cands_rpfp->Pop(1);
|
||||||
}
|
}
|
||||||
gen_cands_rpfp->Pop(1);
|
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
updated_nodes.clear();
|
updated_nodes.clear();
|
||||||
|
@ -1767,6 +1826,7 @@ namespace Duality {
|
||||||
}
|
}
|
||||||
timer_stop("Propagate");
|
timer_stop("Propagate");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/** This class represents a derivation tree. */
|
/** This class represents a derivation tree. */
|
||||||
class DerivationTree {
|
class DerivationTree {
|
||||||
|
@ -2118,6 +2178,8 @@ namespace Duality {
|
||||||
|
|
||||||
hash_map<Node *, expr> updates;
|
hash_map<Node *, expr> updates;
|
||||||
|
|
||||||
|
int restart_interval;
|
||||||
|
|
||||||
DerivationTreeSlow(Duality *_duality, RPFP *rpfp, Reporter *_reporter, Heuristic *_heuristic, bool _full_expand)
|
DerivationTreeSlow(Duality *_duality, RPFP *rpfp, Reporter *_reporter, Heuristic *_heuristic, bool _full_expand)
|
||||||
: DerivationTree(_duality, rpfp, _reporter, _heuristic, _full_expand) {
|
: DerivationTree(_duality, rpfp, _reporter, _heuristic, _full_expand) {
|
||||||
stack.push_back(stack_entry());
|
stack.push_back(stack_entry());
|
||||||
|
@ -2126,6 +2188,7 @@ namespace Duality {
|
||||||
struct DoRestart {};
|
struct DoRestart {};
|
||||||
|
|
||||||
virtual bool Build(){
|
virtual bool Build(){
|
||||||
|
restart_interval = 3;
|
||||||
while (true) {
|
while (true) {
|
||||||
try {
|
try {
|
||||||
return BuildMain();
|
return BuildMain();
|
||||||
|
@ -2136,15 +2199,47 @@ namespace Duality {
|
||||||
while(stack.size() > 1)
|
while(stack.size() > 1)
|
||||||
PopLevel();
|
PopLevel();
|
||||||
reporter->Message("restarted");
|
reporter->Message("restarted");
|
||||||
|
restart_interval += 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// When we check, try to use the same children that were used in the
|
||||||
|
// previous counterexample.
|
||||||
|
check_result Check(){
|
||||||
|
#if 0
|
||||||
|
std::vector<Node *> posnodes, negnodes;
|
||||||
|
std::vector<Node *> &expansions = stack.back().expansions;
|
||||||
|
for(unsigned i = 0; i < expansions.size(); i++){
|
||||||
|
Node *node = expansions[i];
|
||||||
|
std::vector<Node*> &chs = node->Outgoing->Children;
|
||||||
|
for(unsigned j = 0; j < chs.size(); j++){
|
||||||
|
Node *ch = chs[j];
|
||||||
|
int use = heuristic->UseNode(ch);
|
||||||
|
if(use == 1)
|
||||||
|
posnodes.push_back(ch);
|
||||||
|
else if (use == -1)
|
||||||
|
negnodes.push_back(ch);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if(!(posnodes.empty() && negnodes.empty())){
|
||||||
|
check_result res = tree->CheckWithConstrainedNodes(posnodes,negnodes);
|
||||||
|
if(res != unsat){
|
||||||
|
reporter->Message("matched previous counterexample");
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
return tree->Check(top);
|
||||||
|
}
|
||||||
|
|
||||||
bool BuildMain(){
|
bool BuildMain(){
|
||||||
|
|
||||||
stack.back().level = tree->slvr().get_scope_level();
|
stack.back().level = tree->slvr().get_scope_level();
|
||||||
bool was_sat = true;
|
bool was_sat = true;
|
||||||
int update_failures = 0;
|
int update_failures = 0;
|
||||||
|
int total_updates = 0;
|
||||||
|
|
||||||
while (true)
|
while (true)
|
||||||
{
|
{
|
||||||
|
@ -2156,7 +2251,7 @@ namespace Duality {
|
||||||
reporter->Depth(stack.size());
|
reporter->Depth(stack.size());
|
||||||
|
|
||||||
// res = tree->Solve(top, 1); // incremental solve, keep interpolants for one pop
|
// res = tree->Solve(top, 1); // incremental solve, keep interpolants for one pop
|
||||||
check_result foo = tree->Check(top);
|
check_result foo = Check();
|
||||||
res = foo == unsat ? l_false : l_true;
|
res = foo == unsat ? l_false : l_true;
|
||||||
|
|
||||||
if (res == l_false) {
|
if (res == l_false) {
|
||||||
|
@ -2168,32 +2263,50 @@ namespace Duality {
|
||||||
int update_count = 0;
|
int update_count = 0;
|
||||||
for(unsigned i = 0; i < expansions.size(); i++){
|
for(unsigned i = 0; i < expansions.size(); i++){
|
||||||
Node *node = expansions[i];
|
Node *node = expansions[i];
|
||||||
tree->SolveSingleNode(top,node);
|
|
||||||
#ifdef NO_GENERALIZE
|
|
||||||
node->Annotation.Formula = tree->RemoveRedundancy(node->Annotation.Formula).simplify();
|
|
||||||
#else
|
|
||||||
try {
|
try {
|
||||||
|
tree->SolveSingleNode(top,node);
|
||||||
|
#ifdef NO_GENERALIZE
|
||||||
|
node->Annotation.Formula = tree->RemoveRedundancy(node->Annotation.Formula).simplify();
|
||||||
|
#else
|
||||||
if(expansions.size() == 1 && NodeTooComplicated(node))
|
if(expansions.size() == 1 && NodeTooComplicated(node))
|
||||||
SimplifyNode(node);
|
SimplifyNode(node);
|
||||||
else
|
else
|
||||||
node->Annotation.Formula = tree->RemoveRedundancy(node->Annotation.Formula).simplify();
|
node->Annotation.Formula = tree->RemoveRedundancy(node->Annotation.Formula).simplify();
|
||||||
Generalize(node);
|
Generalize(node);
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
catch(const RPFP::Bad &){
|
catch(const RPFP::Bad &){
|
||||||
// bad interpolants can get us here
|
// bad interpolants can get us here
|
||||||
throw DoRestart();
|
throw DoRestart();
|
||||||
}
|
}
|
||||||
#endif
|
catch(char const *msg){
|
||||||
if(RecordUpdate(node))
|
// bad interpolants can get us here
|
||||||
|
reporter->Message(std::string("interpolation failure:") + msg);
|
||||||
|
throw DoRestart();
|
||||||
|
}
|
||||||
|
if(RecordUpdate(node)){
|
||||||
update_count++;
|
update_count++;
|
||||||
|
total_updates++;
|
||||||
|
}
|
||||||
else
|
else
|
||||||
heuristic->Update(node->map); // make it less likely to expand this node in future
|
heuristic->Update(node->map); // make it less likely to expand this node in future
|
||||||
}
|
}
|
||||||
|
#if 1
|
||||||
|
if(duality->EnableRestarts)
|
||||||
|
if(total_updates >= restart_interval)
|
||||||
|
throw DoRestart();
|
||||||
|
#endif
|
||||||
if(update_count == 0){
|
if(update_count == 0){
|
||||||
if(was_sat){
|
if(was_sat){
|
||||||
update_failures++;
|
update_failures++;
|
||||||
if(update_failures > 10)
|
if(update_failures > 10){
|
||||||
throw Incompleteness();
|
for(unsigned i = 0; i < expansions.size(); i++){
|
||||||
|
Node *node = expansions[i];
|
||||||
|
node->map->Annotation.SetFull();
|
||||||
|
reporter->Message("incompleteness: cleared annotation");
|
||||||
|
}
|
||||||
|
throw DoRestart();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
reporter->Message("backtracked without learning");
|
reporter->Message("backtracked without learning");
|
||||||
}
|
}
|
||||||
|
@ -2234,6 +2347,10 @@ namespace Duality {
|
||||||
tree->Push();
|
tree->Push();
|
||||||
std::vector<Node *> &expansions = stack.back().expansions;
|
std::vector<Node *> &expansions = stack.back().expansions;
|
||||||
#ifndef NO_DECISIONS
|
#ifndef NO_DECISIONS
|
||||||
|
#if 0
|
||||||
|
if(expansions.size() > 0)
|
||||||
|
tree->GreedyReduceNodes(expansions[0]->Outgoing->Children); // try to reduce number of children
|
||||||
|
#endif
|
||||||
for(unsigned i = 0; i < expansions.size(); i++){
|
for(unsigned i = 0; i < expansions.size(); i++){
|
||||||
tree->FixCurrentState(expansions[i]->Outgoing);
|
tree->FixCurrentState(expansions[i]->Outgoing);
|
||||||
}
|
}
|
||||||
|
@ -2253,15 +2370,42 @@ namespace Duality {
|
||||||
if(ExpandSomeNodes(false,expand_max))
|
if(ExpandSomeNodes(false,expand_max))
|
||||||
continue;
|
continue;
|
||||||
tree->Pop(1);
|
tree->Pop(1);
|
||||||
|
node_order.clear();
|
||||||
while(stack.size() > 1){
|
while(stack.size() > 1){
|
||||||
tree->Pop(1);
|
tree->Pop(1);
|
||||||
|
std::vector<Node *> &expansions = stack.back().expansions;
|
||||||
|
for(unsigned i = 0; i < expansions.size(); i++)
|
||||||
|
node_order.push_back(expansions[i]);
|
||||||
stack.pop_back();
|
stack.pop_back();
|
||||||
}
|
}
|
||||||
|
#if 0
|
||||||
|
Reduce();
|
||||||
|
#endif
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::vector<Node *> node_order;
|
||||||
|
|
||||||
|
void Reduce(){
|
||||||
|
tree->Push();
|
||||||
|
// tree->AssertNode(top); // assert the negation of the top-level spec
|
||||||
|
for(int i = node_order.size()-1; i >= 0; --i){
|
||||||
|
Edge *edge = node_order[i]->Outgoing;
|
||||||
|
if(edge){
|
||||||
|
for(unsigned j = 0; j < edge->Children.size(); j++){
|
||||||
|
Node *ch = edge->Children[j];
|
||||||
|
if(!ch->Outgoing)
|
||||||
|
ch->Annotation.SetEmpty();
|
||||||
|
}
|
||||||
|
tree->AssertEdge(edge,0,true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
tree->GreedyReduceNodes(node_order); // try to reduce the counterexample size
|
||||||
|
tree->Pop(1);
|
||||||
|
}
|
||||||
|
|
||||||
void PopLevel(){
|
void PopLevel(){
|
||||||
std::vector<Node *> &expansions = stack.back().expansions;
|
std::vector<Node *> &expansions = stack.back().expansions;
|
||||||
tree->Pop(1);
|
tree->Pop(1);
|
||||||
|
@ -2860,17 +3004,7 @@ namespace Duality {
|
||||||
return name;
|
return name;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void ChooseExpand(const std::set<RPFP::Node *> &choices, std::set<RPFP::Node *> &best, bool high_priority, bool best_only){
|
Node *MatchNode(Node *node){
|
||||||
if(!high_priority || !old_cex.get_tree()){
|
|
||||||
Heuristic::ChooseExpand(choices,best,false);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
// first, try to match the derivatino tree nodes to the old cex
|
|
||||||
std::set<Node *> matched, unmatched;
|
|
||||||
for(std::set<Node *>::iterator it = choices.begin(), en = choices.end(); it != en; ++it){
|
|
||||||
Node *node = (*it);
|
|
||||||
if(cex_map.empty())
|
|
||||||
cex_map[node] = old_cex.get_root(); // match the root nodes
|
|
||||||
if(cex_map.find(node) == cex_map.end()){ // try to match an unmatched node
|
if(cex_map.find(node) == cex_map.end()){ // try to match an unmatched node
|
||||||
Node *parent = node->Incoming[0]->Parent; // assumes we are a tree!
|
Node *parent = node->Incoming[0]->Parent; // assumes we are a tree!
|
||||||
if(cex_map.find(parent) == cex_map.end())
|
if(cex_map.find(parent) == cex_map.end())
|
||||||
|
@ -2893,7 +3027,30 @@ namespace Duality {
|
||||||
cex_map[chs[i]] = 0;
|
cex_map[chs[i]] = 0;
|
||||||
}
|
}
|
||||||
matching_done:
|
matching_done:
|
||||||
Node *old_node = cex_map[node];
|
return cex_map[node];
|
||||||
|
}
|
||||||
|
|
||||||
|
int UseNode(Node *node){
|
||||||
|
if (!old_cex.get_tree())
|
||||||
|
return 0;
|
||||||
|
Node *old_node = MatchNode(node);
|
||||||
|
if(!old_node)
|
||||||
|
return 0;
|
||||||
|
return old_cex.get_tree()->Empty(old_node) ? -1 : 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void ChooseExpand(const std::set<RPFP::Node *> &choices, std::set<RPFP::Node *> &best, bool high_priority, bool best_only){
|
||||||
|
if(cex_map.empty())
|
||||||
|
cex_map[*(choices.begin())] = old_cex.get_root(); // match the root nodes
|
||||||
|
if(!high_priority || !old_cex.get_tree()){
|
||||||
|
Heuristic::ChooseExpand(choices,best,false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
// first, try to match the derivatino tree nodes to the old cex
|
||||||
|
std::set<Node *> matched, unmatched;
|
||||||
|
for(std::set<Node *>::iterator it = choices.begin(), en = choices.end(); it != en; ++it){
|
||||||
|
Node *node = (*it);
|
||||||
|
Node *old_node = MatchNode(node);
|
||||||
if(!old_node)
|
if(!old_node)
|
||||||
unmatched.insert(node);
|
unmatched.insert(node);
|
||||||
else if(old_cex.get_tree()->Empty(old_node))
|
else if(old_cex.get_tree()->Empty(old_node))
|
||||||
|
@ -3168,8 +3325,233 @@ namespace Duality {
|
||||||
|
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
class DualityDepthBounded : public Solver {
|
||||||
|
|
||||||
|
Duality *duality;
|
||||||
|
context &ctx; // Z3 context
|
||||||
|
solver &slvr; // Z3 solver
|
||||||
|
|
||||||
|
public:
|
||||||
|
DualityDepthBounded(RPFP *_rpfp) :
|
||||||
|
ctx(_rpfp->ctx),
|
||||||
|
slvr(_rpfp->slvr()){
|
||||||
|
rpfp = _rpfp;
|
||||||
|
DepthBoundRPFP();
|
||||||
|
duality = alloc(Duality,drpfp);
|
||||||
|
}
|
||||||
|
|
||||||
|
~DualityDepthBounded(){
|
||||||
|
dealloc(duality);
|
||||||
|
delete drpfp;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool Solve(){
|
||||||
|
int depth_bound = 10;
|
||||||
|
bool res;
|
||||||
|
SetMaxDepthRPFP(depth_bound);
|
||||||
|
duality->PreSolve();
|
||||||
|
while(true){
|
||||||
|
res = duality->SolveMain();
|
||||||
|
if(!res || GetSolution())
|
||||||
|
break;
|
||||||
|
depth_bound++;
|
||||||
|
SetMaxDepthRPFP(depth_bound);
|
||||||
|
res = duality->RecheckBounds();
|
||||||
|
if(!res)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
duality->PostSolve();
|
||||||
|
if(!res)
|
||||||
|
ConvertCex();
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
Counterexample &GetCounterexample(){
|
||||||
|
return cex;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool SetOption(const std::string &option, const std::string &value){
|
||||||
|
return duality->SetOption(option,value);
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual void LearnFrom(Solver *old_solver){
|
||||||
|
DualityDepthBounded *old = dynamic_cast<DualityDepthBounded *>(old_solver);
|
||||||
|
if(old){
|
||||||
|
duality->LearnFrom(old->duality);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
bool IsResultRecursionBounded(){
|
||||||
|
return duality->IsResultRecursionBounded();
|
||||||
|
}
|
||||||
|
|
||||||
|
void Cancel(){
|
||||||
|
duality->Cancel();
|
||||||
|
}
|
||||||
|
|
||||||
|
typedef RPFP::Node Node;
|
||||||
|
typedef RPFP::Edge Edge;
|
||||||
|
RPFP *rpfp, *drpfp;
|
||||||
|
hash_map<Node *, Node *> db_map, db_rev_map;
|
||||||
|
hash_map<Edge *, Edge *> db_edge_rev_map;
|
||||||
|
std::vector<expr> db_saved_bounds;
|
||||||
|
Counterexample cex;
|
||||||
|
|
||||||
|
expr AddParamToRels(hash_map<ast,expr> &memo, hash_map<func_decl,func_decl> &rmap, const expr &p, const expr &t) {
|
||||||
|
std::pair<ast,expr> foo(t,expr(ctx));
|
||||||
|
std::pair<hash_map<ast,expr>::iterator, bool> bar = memo.insert(foo);
|
||||||
|
expr &res = bar.first->second;
|
||||||
|
if(!bar.second) return res;
|
||||||
|
|
||||||
|
if (t.is_app())
|
||||||
|
{
|
||||||
|
func_decl f = t.decl();
|
||||||
|
std::vector<expr> args;
|
||||||
|
int nargs = t.num_args();
|
||||||
|
for(int i = 0; i < nargs; i++)
|
||||||
|
args.push_back(AddParamToRels(memo, rmap, p, t.arg(i)));
|
||||||
|
hash_map<func_decl,func_decl>::iterator rit = rmap.find(f);
|
||||||
|
if(rit != rmap.end()){
|
||||||
|
args.push_back(p);
|
||||||
|
res = (rit->second)(args);
|
||||||
|
res = ctx.make(And,res,ctx.make(Geq,p,ctx.int_val(0)));
|
||||||
|
}
|
||||||
|
else
|
||||||
|
res = f(args);
|
||||||
|
}
|
||||||
|
else if (t.is_quantifier())
|
||||||
|
{
|
||||||
|
expr body = AddParamToRels(memo, rmap, p, t.body());
|
||||||
|
res = clone_quantifier(t, body);
|
||||||
|
}
|
||||||
|
else res = t;
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void DepthBoundRPFP(){
|
||||||
|
drpfp = new RPFP(rpfp->ls);
|
||||||
|
expr dvar = ctx.int_const("@depth");
|
||||||
|
expr dmax = ctx.int_const("@depth_max");
|
||||||
|
for(unsigned i = 0; i < rpfp->nodes.size(); i++){
|
||||||
|
Node *node = rpfp->nodes[i];
|
||||||
|
std::vector<sort> arg_sorts;
|
||||||
|
const std::vector<expr> ¶ms = node->Annotation.IndParams;
|
||||||
|
for(unsigned j = 0; j < params.size(); j++)
|
||||||
|
arg_sorts.push_back(params[j].get_sort());
|
||||||
|
arg_sorts.push_back(ctx.int_sort());
|
||||||
|
std::string new_name = std::string("@db@") + node->Name.name().str();
|
||||||
|
func_decl f = ctx.function(new_name.c_str(),arg_sorts.size(), &arg_sorts[0],ctx.bool_sort());
|
||||||
|
std::vector<expr> args = params;
|
||||||
|
args.push_back(dvar);
|
||||||
|
expr pat = f(args);
|
||||||
|
Node *dnode = drpfp->CreateNode(pat);
|
||||||
|
db_map[node] = dnode;
|
||||||
|
db_rev_map[dnode] = node;
|
||||||
|
expr bound_fmla = node->Bound.Formula;
|
||||||
|
if(!eq(bound_fmla,ctx.bool_val(true))){
|
||||||
|
bound_fmla = implies(dvar == dmax,bound_fmla);
|
||||||
|
dnode->Bound.Formula = bound_fmla;
|
||||||
|
}
|
||||||
|
db_saved_bounds.push_back(bound_fmla);
|
||||||
|
// dnode->Annotation.Formula = ctx.make(And,node->Annotation.Formula,ctx.make(Geq,dvar,ctx.int_val(0)));
|
||||||
|
}
|
||||||
|
for(unsigned i = 0; i < rpfp->edges.size(); i++){
|
||||||
|
Edge *edge = rpfp->edges[i];
|
||||||
|
std::vector<Node *> new_children;
|
||||||
|
std::vector<func_decl> new_relparams;
|
||||||
|
hash_map<func_decl,func_decl> rmap;
|
||||||
|
for(unsigned j = 0; j < edge->Children.size(); j++){
|
||||||
|
Node *ch = edge->Children[j];
|
||||||
|
Node *nch = db_map[ch];
|
||||||
|
func_decl f = nch->Name;
|
||||||
|
func_decl sf = drpfp->NumberPred(f,j);
|
||||||
|
new_children.push_back(nch);
|
||||||
|
new_relparams.push_back(sf);
|
||||||
|
rmap[edge->F.RelParams[j]] = sf;
|
||||||
|
}
|
||||||
|
std::vector<expr> new_indparams = edge->F.IndParams;
|
||||||
|
new_indparams.push_back(dvar);
|
||||||
|
hash_map<ast,expr> memo;
|
||||||
|
expr new_fmla = AddParamToRels(memo,rmap,ctx.make(Sub,dvar,ctx.int_val(1)),edge->F.Formula);
|
||||||
|
RPFP::Transformer new_t = drpfp->CreateTransformer(new_relparams,new_indparams,new_fmla);
|
||||||
|
Node *new_parent = db_map[edge->Parent];
|
||||||
|
db_edge_rev_map[drpfp->CreateEdge(new_parent,new_t,new_children)] = edge;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void SetMaxDepthRPFP(int depth){
|
||||||
|
hash_map<ast,expr> subst;
|
||||||
|
expr dmax = ctx.int_const("@depth_max");
|
||||||
|
subst[dmax] = ctx.int_val(depth);
|
||||||
|
for(unsigned i = 0; i < drpfp->nodes.size(); i++){
|
||||||
|
Node *node = drpfp->nodes[i];
|
||||||
|
expr fmla = db_saved_bounds[i];
|
||||||
|
fmla = drpfp->SubstRec(subst,fmla);
|
||||||
|
node->Bound.Formula = fmla;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void ConvertCex(){
|
||||||
|
cex.clear();
|
||||||
|
RPFP *tree = new RPFP(rpfp->ls);
|
||||||
|
Node *root;
|
||||||
|
Counterexample &dctx = duality->GetCounterexample();
|
||||||
|
hash_map<Node *, Node *> ctx_node_map;
|
||||||
|
for(unsigned i = 0; i < dctx.get_tree()->nodes.size(); i++){
|
||||||
|
Node *dnode = dctx.get_tree()->nodes[i];
|
||||||
|
Node *onode = db_rev_map[dnode->map->map];
|
||||||
|
Node *node = tree->CloneNode(onode);
|
||||||
|
node->number = dnode->number; // numbers have to match for model to make sense
|
||||||
|
ctx_node_map[dnode] = node;
|
||||||
|
if(dnode == dctx.get_root())
|
||||||
|
root = node;
|
||||||
|
}
|
||||||
|
for(unsigned i = 0; i < dctx.get_tree()->edges.size(); i++){
|
||||||
|
Edge *dedge = dctx.get_tree()->edges[i];
|
||||||
|
Edge *oedge = db_edge_rev_map[dedge->map];
|
||||||
|
Node *parent = ctx_node_map[dedge->Parent];
|
||||||
|
std::vector<Node *> chs;
|
||||||
|
for(unsigned j = 0; j < dedge->Children.size(); j++)
|
||||||
|
chs.push_back(ctx_node_map[dedge->Children[j]]);
|
||||||
|
Edge *edge = tree->CreateEdge(parent,oedge->F,chs);
|
||||||
|
edge->number = dedge->number; // numbers have to match for model to make sense
|
||||||
|
edge->map = oedge;
|
||||||
|
}
|
||||||
|
tree->dualModel = dctx.get_tree()->dualModel;
|
||||||
|
cex.set(tree,root);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool GetSolution(){
|
||||||
|
for(unsigned i = 0; i < rpfp->nodes.size(); i++)
|
||||||
|
if(!drpfp->nodes[i]->Annotation.SubsetEq(rpfp->nodes[i]->Bound))
|
||||||
|
return false;
|
||||||
|
expr dvar = ctx.int_const("@depth");
|
||||||
|
hash_map<ast,expr> subst;
|
||||||
|
subst[dvar] = ctx.int_val(INT_MAX);
|
||||||
|
for(unsigned i = 0; i < rpfp->nodes.size(); i++){
|
||||||
|
expr fmla = drpfp->nodes[i]->Annotation.Formula;
|
||||||
|
fmla = drpfp->SubstRec(subst,fmla);
|
||||||
|
fmla = fmla.simplify();
|
||||||
|
rpfp->nodes[i]->Annotation.Formula = fmla;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
void UndoDepthBoundRPFP(){
|
||||||
|
#if 0
|
||||||
|
if(cex.get_tree()){
|
||||||
|
// here, need to map the cex back...
|
||||||
|
}
|
||||||
|
// also need to map the proof back, but we don't...
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
Solver *Solver::Create(const std::string &solver_class, RPFP *rpfp){
|
Solver *Solver::Create(const std::string &solver_class, RPFP *rpfp){
|
||||||
Duality *s = alloc(Duality,rpfp);
|
// Solver *s = alloc(DualityDepthBounded,rpfp);
|
||||||
|
Solver *s = alloc(Duality,rpfp);
|
||||||
return s;
|
return s;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -513,3 +513,26 @@ void interpolation_options_struct::apply(iz3base &b){
|
||||||
b.set_option((*it).first,(*it).second);
|
b.set_option((*it).first,(*it).second);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// On linux and mac, unlimit stack space so we get recursion
|
||||||
|
|
||||||
|
#if defined(_WINDOWS) || defined(_CYGWIN)
|
||||||
|
|
||||||
|
#else
|
||||||
|
|
||||||
|
#include <sys/time.h>
|
||||||
|
#include <sys/resource.h>
|
||||||
|
|
||||||
|
class iz3stack_unlimiter {
|
||||||
|
public:
|
||||||
|
iz3stack_unlimiter() {
|
||||||
|
struct rlimit rl = {RLIM_INFINITY, RLIM_INFINITY};
|
||||||
|
setrlimit(RLIMIT_STACK, &rl);
|
||||||
|
// nothing to be done if above fails
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// initializing this will unlimit stack
|
||||||
|
|
||||||
|
iz3stack_unlimiter the_iz3stack_unlimiter;
|
||||||
|
|
||||||
|
#endif
|
||||||
|
|
|
@ -2,4 +2,5 @@ def_module_params('interp',
|
||||||
description='interpolation parameters',
|
description='interpolation parameters',
|
||||||
export=True,
|
export=True,
|
||||||
params=(('profile', BOOL, False, '(INTERP) profile interpolation'),
|
params=(('profile', BOOL, False, '(INTERP) profile interpolation'),
|
||||||
|
('check', BOOL, False, '(INTERP) check interpolants'),
|
||||||
))
|
))
|
||||||
|
|
|
@ -3078,6 +3078,11 @@ public:
|
||||||
m().dec_ref(add_pos);
|
m().dec_ref(add_pos);
|
||||||
m().dec_ref(rewrite_A);
|
m().dec_ref(rewrite_A);
|
||||||
m().dec_ref(rewrite_B);
|
m().dec_ref(rewrite_B);
|
||||||
|
m().dec_ref(normal_step);
|
||||||
|
m().dec_ref(normal_chain);
|
||||||
|
m().dec_ref(normal);
|
||||||
|
m().dec_ref(sforall);
|
||||||
|
m().dec_ref(sexists);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -245,6 +245,7 @@ namespace datalog {
|
||||||
m_rule_fmls_head = 0;
|
m_rule_fmls_head = 0;
|
||||||
m_rule_fmls.reset();
|
m_rule_fmls.reset();
|
||||||
m_rule_names.reset();
|
m_rule_names.reset();
|
||||||
|
m_rule_bounds.reset();
|
||||||
m_argument_var_names.reset();
|
m_argument_var_names.reset();
|
||||||
m_preds.reset();
|
m_preds.reset();
|
||||||
m_preds_by_name.reset();
|
m_preds_by_name.reset();
|
||||||
|
@ -440,9 +441,10 @@ namespace datalog {
|
||||||
return new_pred;
|
return new_pred;
|
||||||
}
|
}
|
||||||
|
|
||||||
void context::add_rule(expr* rl, symbol const& name) {
|
void context::add_rule(expr* rl, symbol const& name, unsigned bound) {
|
||||||
m_rule_fmls.push_back(rl);
|
m_rule_fmls.push_back(rl);
|
||||||
m_rule_names.push_back(name);
|
m_rule_names.push_back(name);
|
||||||
|
m_rule_bounds.push_back(bound);
|
||||||
}
|
}
|
||||||
|
|
||||||
void context::flush_add_rules() {
|
void context::flush_add_rules() {
|
||||||
|
@ -859,6 +861,9 @@ namespace datalog {
|
||||||
flush_add_rules();
|
flush_add_rules();
|
||||||
break;
|
break;
|
||||||
case DUALITY_ENGINE:
|
case DUALITY_ENGINE:
|
||||||
|
// this lets us use duality with SAS 2013 abstraction
|
||||||
|
if(quantify_arrays())
|
||||||
|
flush_add_rules();
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE();
|
||||||
|
@ -979,12 +984,13 @@ namespace datalog {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void context::get_raw_rule_formulas(expr_ref_vector& rules, svector<symbol>& names){
|
void context::get_raw_rule_formulas(expr_ref_vector& rules, svector<symbol>& names, vector<unsigned> &bounds){
|
||||||
for (unsigned i = 0; i < m_rule_fmls.size(); ++i) {
|
for (unsigned i = 0; i < m_rule_fmls.size(); ++i) {
|
||||||
expr_ref r = bind_vars(m_rule_fmls[i].get(), true);
|
expr_ref r = bind_vars(m_rule_fmls[i].get(), true);
|
||||||
rules.push_back(r.get());
|
rules.push_back(r.get());
|
||||||
// rules.push_back(m_rule_fmls[i].get());
|
// rules.push_back(m_rule_fmls[i].get());
|
||||||
names.push_back(m_rule_names[i]);
|
names.push_back(m_rule_names[i]);
|
||||||
|
bounds.push_back(m_rule_bounds[i]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1002,6 +1008,7 @@ namespace datalog {
|
||||||
m_rule_names[i] = m_rule_names.back();
|
m_rule_names[i] = m_rule_names.back();
|
||||||
m_rule_fmls.pop_back();
|
m_rule_fmls.pop_back();
|
||||||
m_rule_names.pop_back();
|
m_rule_names.pop_back();
|
||||||
|
m_rule_bounds.pop_back();
|
||||||
--i;
|
--i;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -195,6 +195,7 @@ namespace datalog {
|
||||||
unsigned m_rule_fmls_head;
|
unsigned m_rule_fmls_head;
|
||||||
expr_ref_vector m_rule_fmls;
|
expr_ref_vector m_rule_fmls;
|
||||||
svector<symbol> m_rule_names;
|
svector<symbol> m_rule_names;
|
||||||
|
vector<unsigned> m_rule_bounds;
|
||||||
expr_ref_vector m_background;
|
expr_ref_vector m_background;
|
||||||
model_converter_ref m_mc;
|
model_converter_ref m_mc;
|
||||||
proof_converter_ref m_pc;
|
proof_converter_ref m_pc;
|
||||||
|
@ -372,7 +373,7 @@ namespace datalog {
|
||||||
rule_set & get_rules() { flush_add_rules(); return m_rule_set; }
|
rule_set & get_rules() { flush_add_rules(); return m_rule_set; }
|
||||||
|
|
||||||
void get_rules_as_formulas(expr_ref_vector& fmls, expr_ref_vector& qs, svector<symbol>& names);
|
void get_rules_as_formulas(expr_ref_vector& fmls, expr_ref_vector& qs, svector<symbol>& names);
|
||||||
void get_raw_rule_formulas(expr_ref_vector& fmls, svector<symbol>& names);
|
void get_raw_rule_formulas(expr_ref_vector& fmls, svector<symbol>& names, vector<unsigned> &bounds);
|
||||||
|
|
||||||
void add_fact(app * head);
|
void add_fact(app * head);
|
||||||
void add_fact(func_decl * pred, const relation_fact & fact);
|
void add_fact(func_decl * pred, const relation_fact & fact);
|
||||||
|
@ -389,7 +390,7 @@ namespace datalog {
|
||||||
/**
|
/**
|
||||||
Method exposed from API for adding rules.
|
Method exposed from API for adding rules.
|
||||||
*/
|
*/
|
||||||
void add_rule(expr* rl, symbol const& name);
|
void add_rule(expr* rl, symbol const& name, unsigned bound = UINT_MAX);
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -142,6 +142,7 @@ def_module_params('fixedpoint',
|
||||||
('xform.instantiate_quantifiers', BOOL, False,
|
('xform.instantiate_quantifiers', BOOL, False,
|
||||||
"instantiate quantified Horn clauses using E-matching heuristic"),
|
"instantiate quantified Horn clauses using E-matching heuristic"),
|
||||||
('xform.coalesce_rules', BOOL, False, "coalesce rules"),
|
('xform.coalesce_rules', BOOL, False, "coalesce rules"),
|
||||||
|
('duality.enable_restarts', BOOL, False, 'DUALITY: enable restarts'),
|
||||||
))
|
))
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -37,6 +37,7 @@ Revision History:
|
||||||
#include "fixedpoint_params.hpp"
|
#include "fixedpoint_params.hpp"
|
||||||
#include "used_vars.h"
|
#include "used_vars.h"
|
||||||
#include "func_decl_dependencies.h"
|
#include "func_decl_dependencies.h"
|
||||||
|
#include "dl_transforms.h"
|
||||||
|
|
||||||
// template class symbol_table<family_id>;
|
// template class symbol_table<family_id>;
|
||||||
|
|
||||||
|
@ -155,8 +156,41 @@ lbool dl_interface::query(::expr * query) {
|
||||||
|
|
||||||
expr_ref_vector rules(m_ctx.get_manager());
|
expr_ref_vector rules(m_ctx.get_manager());
|
||||||
svector< ::symbol> names;
|
svector< ::symbol> names;
|
||||||
|
vector<unsigned> bounds;
|
||||||
// m_ctx.get_rules_as_formulas(rules, names);
|
// m_ctx.get_rules_as_formulas(rules, names);
|
||||||
m_ctx.get_raw_rule_formulas(rules, names);
|
|
||||||
|
|
||||||
|
// If using SAS 2013 abstractiion, we need to perform some transforms
|
||||||
|
expr_ref query_ref(m_ctx.get_manager());
|
||||||
|
if(m_ctx.quantify_arrays()){
|
||||||
|
datalog::rule_manager& rm = m_ctx.get_rule_manager();
|
||||||
|
rm.mk_query(query, m_ctx.get_rules());
|
||||||
|
apply_default_transformation(m_ctx);
|
||||||
|
datalog::rule_set &rs = m_ctx.get_rules();
|
||||||
|
if(m_ctx.get_rules().get_output_predicates().empty())
|
||||||
|
query_ref = m_ctx.get_manager().mk_false();
|
||||||
|
else {
|
||||||
|
func_decl_ref query_pred(m_ctx.get_manager());
|
||||||
|
query_pred = m_ctx.get_rules().get_output_predicate();
|
||||||
|
ptr_vector<sort> sorts;
|
||||||
|
unsigned nargs = query_pred.get()->get_arity();
|
||||||
|
expr_ref_vector vars(m_ctx.get_manager());
|
||||||
|
for(unsigned i = 0; i < nargs; i++){
|
||||||
|
::sort *s = query_pred.get()->get_domain(i);
|
||||||
|
vars.push_back(m_ctx.get_manager().mk_var(nargs-1-i,s));
|
||||||
|
}
|
||||||
|
query_ref = m_ctx.get_manager().mk_app(query_pred.get(),nargs,vars.c_ptr());
|
||||||
|
query = query_ref.get();
|
||||||
|
}
|
||||||
|
unsigned nrules = rs.get_num_rules();
|
||||||
|
for(unsigned i = 0; i < nrules; i++){
|
||||||
|
expr_ref f(m_ctx.get_manager());
|
||||||
|
rm.to_formula(*rs.get_rule(i), f);
|
||||||
|
rules.push_back(f);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
m_ctx.get_raw_rule_formulas(rules, names, bounds);
|
||||||
|
|
||||||
// get all the rules as clauses
|
// get all the rules as clauses
|
||||||
std::vector<expr> &clauses = _d->clauses;
|
std::vector<expr> &clauses = _d->clauses;
|
||||||
|
@ -200,6 +234,7 @@ lbool dl_interface::query(::expr * query) {
|
||||||
expr qc = implies(q,_d->ctx.bool_val(false));
|
expr qc = implies(q,_d->ctx.bool_val(false));
|
||||||
qc = _d->ctx.make_quant(Forall,b_sorts,b_names,qc);
|
qc = _d->ctx.make_quant(Forall,b_sorts,b_names,qc);
|
||||||
clauses.push_back(qc);
|
clauses.push_back(qc);
|
||||||
|
bounds.push_back(UINT_MAX);
|
||||||
|
|
||||||
// get the background axioms
|
// get the background axioms
|
||||||
unsigned num_asserts = m_ctx.get_num_assertions();
|
unsigned num_asserts = m_ctx.get_num_assertions();
|
||||||
|
@ -243,13 +278,21 @@ lbool dl_interface::query(::expr * query) {
|
||||||
expr c = implies(_d->ctx.bool_val(false),f(args));
|
expr c = implies(_d->ctx.bool_val(false),f(args));
|
||||||
c = _d->ctx.make_quant(Forall,args,c);
|
c = _d->ctx.make_quant(Forall,args,c);
|
||||||
clauses.push_back(c);
|
clauses.push_back(c);
|
||||||
|
bounds.push_back(UINT_MAX);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
unsigned rb = m_ctx.get_params().duality_recursion_bound();
|
||||||
|
std::vector<unsigned> std_bounds;
|
||||||
|
for(unsigned i = 0; i < bounds.size(); i++){
|
||||||
|
unsigned b = bounds[i];
|
||||||
|
if (b == UINT_MAX) b = rb;
|
||||||
|
std_bounds.push_back(b);
|
||||||
|
}
|
||||||
|
|
||||||
// creates 1-1 map between clauses and rpfp edges
|
// creates 1-1 map between clauses and rpfp edges
|
||||||
_d->rpfp->FromClauses(clauses);
|
_d->rpfp->FromClauses(clauses,&std_bounds);
|
||||||
|
|
||||||
// populate the edge-to-clause map
|
// populate the edge-to-clause map
|
||||||
for(unsigned i = 0; i < _d->rpfp->edges.size(); ++i)
|
for(unsigned i = 0; i < _d->rpfp->edges.size(); ++i)
|
||||||
|
@ -271,11 +314,13 @@ lbool dl_interface::query(::expr * query) {
|
||||||
rs->SetOption("stratified_inlining",m_ctx.get_params().duality_stratified_inlining() ? "1" : "0");
|
rs->SetOption("stratified_inlining",m_ctx.get_params().duality_stratified_inlining() ? "1" : "0");
|
||||||
rs->SetOption("batch_expand",m_ctx.get_params().duality_batch_expand() ? "1" : "0");
|
rs->SetOption("batch_expand",m_ctx.get_params().duality_batch_expand() ? "1" : "0");
|
||||||
rs->SetOption("conjecture_file",m_ctx.get_params().duality_conjecture_file());
|
rs->SetOption("conjecture_file",m_ctx.get_params().duality_conjecture_file());
|
||||||
unsigned rb = m_ctx.get_params().duality_recursion_bound();
|
rs->SetOption("enable_restarts",m_ctx.get_params().duality_enable_restarts() ? "1" : "0");
|
||||||
|
#if 0
|
||||||
if(rb != UINT_MAX){
|
if(rb != UINT_MAX){
|
||||||
std::ostringstream os; os << rb;
|
std::ostringstream os; os << rb;
|
||||||
rs->SetOption("recursion_bound", os.str());
|
rs->SetOption("recursion_bound", os.str());
|
||||||
}
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
// Solve!
|
// Solve!
|
||||||
bool ans;
|
bool ans;
|
||||||
|
|
|
@ -100,7 +100,7 @@ struct dl_context {
|
||||||
dlctx().set_predicate_representation(pred, num_kinds, kinds);
|
dlctx().set_predicate_representation(pred, num_kinds, kinds);
|
||||||
}
|
}
|
||||||
|
|
||||||
void add_rule(expr * rule, symbol const& name) {
|
void add_rule(expr * rule, symbol const& name, unsigned bound) {
|
||||||
init();
|
init();
|
||||||
if (m_collected_cmds) {
|
if (m_collected_cmds) {
|
||||||
expr_ref rl = m_context->bind_vars(rule, true);
|
expr_ref rl = m_context->bind_vars(rule, true);
|
||||||
|
@ -110,7 +110,7 @@ struct dl_context {
|
||||||
m_trail.push(push_back_vector<dl_context, svector<symbol> >(m_collected_cmds->m_names));
|
m_trail.push(push_back_vector<dl_context, svector<symbol> >(m_collected_cmds->m_names));
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
m_context->add_rule(rule, name);
|
m_context->add_rule(rule, name, bound);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -151,19 +151,22 @@ class dl_rule_cmd : public cmd {
|
||||||
mutable unsigned m_arg_idx;
|
mutable unsigned m_arg_idx;
|
||||||
expr* m_t;
|
expr* m_t;
|
||||||
symbol m_name;
|
symbol m_name;
|
||||||
|
unsigned m_bound;
|
||||||
public:
|
public:
|
||||||
dl_rule_cmd(dl_context * dl_ctx):
|
dl_rule_cmd(dl_context * dl_ctx):
|
||||||
cmd("rule"),
|
cmd("rule"),
|
||||||
m_dl_ctx(dl_ctx),
|
m_dl_ctx(dl_ctx),
|
||||||
m_arg_idx(0),
|
m_arg_idx(0),
|
||||||
m_t(0) {}
|
m_t(0),
|
||||||
virtual char const * get_usage() const { return "(forall (q) (=> (and body) head)) :optional-name"; }
|
m_bound(UINT_MAX) {}
|
||||||
|
virtual char const * get_usage() const { return "(forall (q) (=> (and body) head)) :optional-name :optional-recursion-bound"; }
|
||||||
virtual char const * get_descr(cmd_context & ctx) const { return "add a Horn rule."; }
|
virtual char const * get_descr(cmd_context & ctx) const { return "add a Horn rule."; }
|
||||||
virtual unsigned get_arity() const { return VAR_ARITY; }
|
virtual unsigned get_arity() const { return VAR_ARITY; }
|
||||||
virtual cmd_arg_kind next_arg_kind(cmd_context & ctx) const {
|
virtual cmd_arg_kind next_arg_kind(cmd_context & ctx) const {
|
||||||
switch(m_arg_idx) {
|
switch(m_arg_idx) {
|
||||||
case 0: return CPK_EXPR;
|
case 0: return CPK_EXPR;
|
||||||
case 1: return CPK_SYMBOL;
|
case 1: return CPK_SYMBOL;
|
||||||
|
case 2: return CPK_UINT;
|
||||||
default: return CPK_SYMBOL;
|
default: return CPK_SYMBOL;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -173,13 +176,18 @@ public:
|
||||||
}
|
}
|
||||||
virtual void set_next_arg(cmd_context & ctx, symbol const & s) {
|
virtual void set_next_arg(cmd_context & ctx, symbol const & s) {
|
||||||
m_name = s;
|
m_name = s;
|
||||||
|
m_arg_idx++;
|
||||||
|
}
|
||||||
|
virtual void set_next_arg(cmd_context & ctx, unsigned bound) {
|
||||||
|
m_bound = bound;
|
||||||
|
m_arg_idx++;
|
||||||
}
|
}
|
||||||
virtual void reset(cmd_context & ctx) { m_dl_ctx->reset(); prepare(ctx); }
|
virtual void reset(cmd_context & ctx) { m_dl_ctx->reset(); prepare(ctx); }
|
||||||
virtual void prepare(cmd_context& ctx) { m_arg_idx = 0; m_name = symbol::null; }
|
virtual void prepare(cmd_context& ctx) { m_arg_idx = 0; m_name = symbol::null; m_bound = UINT_MAX; }
|
||||||
virtual void finalize(cmd_context & ctx) {
|
virtual void finalize(cmd_context & ctx) {
|
||||||
}
|
}
|
||||||
virtual void execute(cmd_context & ctx) {
|
virtual void execute(cmd_context & ctx) {
|
||||||
m_dl_ctx->add_rule(m_t, m_name);
|
m_dl_ctx->add_rule(m_t, m_name, m_bound);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -48,13 +48,6 @@ namespace datalog {
|
||||||
transf.register_plugin(alloc(datalog::mk_quantifier_abstraction, ctx, 38000));
|
transf.register_plugin(alloc(datalog::mk_quantifier_abstraction, ctx, 38000));
|
||||||
}
|
}
|
||||||
transf.register_plugin(alloc(datalog::mk_quantifier_instantiation, ctx, 37000));
|
transf.register_plugin(alloc(datalog::mk_quantifier_instantiation, ctx, 37000));
|
||||||
transf.register_plugin(alloc(datalog::mk_scale, ctx, 36030));
|
|
||||||
if (ctx.get_params().xform_magic()) {
|
|
||||||
transf.register_plugin(alloc(datalog::mk_magic_symbolic, ctx, 36020));
|
|
||||||
}
|
|
||||||
transf.register_plugin(alloc(datalog::mk_karr_invariants, ctx, 36010));
|
|
||||||
transf.register_plugin(alloc(datalog::mk_array_blast, ctx, 36000));
|
|
||||||
transf.register_plugin(alloc(datalog::mk_bit_blast, ctx, 35000));
|
|
||||||
|
|
||||||
transf.register_plugin(alloc(datalog::mk_subsumption_checker, ctx, 35005));
|
transf.register_plugin(alloc(datalog::mk_subsumption_checker, ctx, 35005));
|
||||||
transf.register_plugin(alloc(datalog::mk_rule_inliner, ctx, 35000));
|
transf.register_plugin(alloc(datalog::mk_rule_inliner, ctx, 35000));
|
||||||
|
@ -75,9 +68,15 @@ namespace datalog {
|
||||||
transf.register_plugin(alloc(datalog::mk_rule_inliner, ctx, 34890));
|
transf.register_plugin(alloc(datalog::mk_rule_inliner, ctx, 34890));
|
||||||
transf.register_plugin(alloc(datalog::mk_subsumption_checker, ctx, 34880));
|
transf.register_plugin(alloc(datalog::mk_subsumption_checker, ctx, 34880));
|
||||||
|
|
||||||
|
transf.register_plugin(alloc(datalog::mk_bit_blast, ctx, 35000));
|
||||||
|
transf.register_plugin(alloc(datalog::mk_karr_invariants, ctx, 36010));
|
||||||
|
transf.register_plugin(alloc(datalog::mk_scale, ctx, 36030));
|
||||||
|
if (!ctx.get_params().xform_quantify_arrays()) {
|
||||||
|
transf.register_plugin(alloc(datalog::mk_array_blast, ctx, 36000));
|
||||||
|
}
|
||||||
|
if (ctx.get_params().xform_magic()) {
|
||||||
|
transf.register_plugin(alloc(datalog::mk_magic_symbolic, ctx, 36020));
|
||||||
|
}
|
||||||
ctx.transform_rules(transf);
|
ctx.transform_rules(transf);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1202,7 +1202,8 @@ namespace opt {
|
||||||
case O_MINIMIZE:
|
case O_MINIMIZE:
|
||||||
case O_MAXIMIZE: {
|
case O_MAXIMIZE: {
|
||||||
inf_eps n = m_optsmt.get_lower(obj.m_index);
|
inf_eps n = m_optsmt.get_lower(obj.m_index);
|
||||||
if (n.get_infinity().is_zero() &&
|
if (m_optsmt.objective_is_model_valid(obj.m_index) &&
|
||||||
|
n.get_infinity().is_zero() &&
|
||||||
n.get_infinitesimal().is_zero() &&
|
n.get_infinitesimal().is_zero() &&
|
||||||
m_model->eval(obj.m_term, val) &&
|
m_model->eval(obj.m_term, val) &&
|
||||||
is_numeral(val, r1)) {
|
is_numeral(val, r1)) {
|
||||||
|
|
|
@ -43,6 +43,7 @@ namespace opt {
|
||||||
m(mgr),
|
m(mgr),
|
||||||
m_dump_benchmarks(false),
|
m_dump_benchmarks(false),
|
||||||
m_fm(fm),
|
m_fm(fm),
|
||||||
|
m_objective_sorts(m),
|
||||||
m_first(true) {
|
m_first(true) {
|
||||||
m_params.updt_params(p);
|
m_params.updt_params(p);
|
||||||
m_params.m_relevancy_lvl = 0;
|
m_params.m_relevancy_lvl = 0;
|
||||||
|
@ -173,13 +174,56 @@ namespace opt {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
\brief maximize the value of objective i in the current state.
|
||||||
|
Return a predicate that blocks the current maximal value.
|
||||||
|
|
||||||
|
The result of 'maximize' is post-processed.
|
||||||
|
When maximization involves shared symbols the model produced
|
||||||
|
by local optimization does not necessarily satisfy combination
|
||||||
|
constraints (it may not be a real model).
|
||||||
|
In this case, the model is post-processed (update_model
|
||||||
|
causes an additional call to final_check to propagate theory equalities
|
||||||
|
when 'has_shared' is true).
|
||||||
|
|
||||||
|
*/
|
||||||
void opt_solver::maximize_objective(unsigned i, expr_ref& blocker) {
|
void opt_solver::maximize_objective(unsigned i, expr_ref& blocker) {
|
||||||
smt::theory_var v = m_objective_vars[i];
|
smt::theory_var v = m_objective_vars[i];
|
||||||
m_objective_values[i] = get_optimizer().maximize(v, blocker);
|
bool has_shared = false;
|
||||||
m_context.get_context().update_model();
|
inf_eps val = get_optimizer().maximize(v, blocker, has_shared);
|
||||||
|
inf_eps val2;
|
||||||
|
m_valid_objectives[i] = true;
|
||||||
|
if (m_context.get_context().update_model(has_shared)) {
|
||||||
|
if (has_shared) {
|
||||||
|
val2 = current_objective_value(i);
|
||||||
|
if (val2 != val) {
|
||||||
|
decrement_value(i, val);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
SASSERT(has_shared);
|
||||||
|
// model is not final. We set the current objective to
|
||||||
|
// close to the optimal (ignoring types).
|
||||||
|
decrement_value(i, val);
|
||||||
|
}
|
||||||
|
m_objective_values[i] = val;
|
||||||
|
|
||||||
TRACE("opt", { model_ref mdl; tout << m_objective_values[i] << "\n";
|
TRACE("opt", { model_ref mdl; tout << m_objective_values[i] << "\n";
|
||||||
get_model(mdl); model_smt2_pp(tout << "update model: ", m, *mdl, 0); });
|
tout << blocker << "\n";
|
||||||
|
get_model(mdl); model_smt2_pp(tout << "update model:\n", m, *mdl, 0); });
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void opt_solver::decrement_value(unsigned i, inf_eps& val) {
|
||||||
|
if (arith_util(m).is_real(m_objective_sorts[i].get())) {
|
||||||
|
val -= inf_eps(inf_rational(rational(0),true));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
val -= inf_eps(inf_rational(rational(1)));
|
||||||
|
}
|
||||||
|
m_valid_objectives[i] = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
void opt_solver::get_unsat_core(ptr_vector<expr> & r) {
|
void opt_solver::get_unsat_core(ptr_vector<expr> & r) {
|
||||||
unsigned sz = m_context.get_unsat_core_size();
|
unsigned sz = m_context.get_unsat_core_size();
|
||||||
|
@ -232,6 +276,8 @@ namespace opt {
|
||||||
smt::theory_var v = get_optimizer().add_objective(term);
|
smt::theory_var v = get_optimizer().add_objective(term);
|
||||||
m_objective_vars.push_back(v);
|
m_objective_vars.push_back(v);
|
||||||
m_objective_values.push_back(inf_eps(rational(-1), inf_rational()));
|
m_objective_values.push_back(inf_eps(rational(-1), inf_rational()));
|
||||||
|
m_objective_sorts.push_back(m.get_sort(term));
|
||||||
|
m_valid_objectives.push_back(true);
|
||||||
return v;
|
return v;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -278,6 +324,8 @@ namespace opt {
|
||||||
void opt_solver::reset_objectives() {
|
void opt_solver::reset_objectives() {
|
||||||
m_objective_vars.reset();
|
m_objective_vars.reset();
|
||||||
m_objective_values.reset();
|
m_objective_values.reset();
|
||||||
|
m_objective_sorts.reset();
|
||||||
|
m_valid_objectives.reset();
|
||||||
}
|
}
|
||||||
|
|
||||||
opt_solver& opt_solver::to_opt(solver& s) {
|
opt_solver& opt_solver::to_opt(solver& s) {
|
||||||
|
|
|
@ -76,6 +76,8 @@ namespace opt {
|
||||||
bool m_objective_enabled;
|
bool m_objective_enabled;
|
||||||
svector<smt::theory_var> m_objective_vars;
|
svector<smt::theory_var> m_objective_vars;
|
||||||
vector<inf_eps> m_objective_values;
|
vector<inf_eps> m_objective_values;
|
||||||
|
sort_ref_vector m_objective_sorts;
|
||||||
|
svector<bool> m_valid_objectives;
|
||||||
bool m_dump_benchmarks;
|
bool m_dump_benchmarks;
|
||||||
static unsigned m_dump_count;
|
static unsigned m_dump_count;
|
||||||
statistics m_stats;
|
statistics m_stats;
|
||||||
|
@ -109,6 +111,9 @@ namespace opt {
|
||||||
void maximize_objectives(expr_ref_vector& blockers);
|
void maximize_objectives(expr_ref_vector& blockers);
|
||||||
inf_eps const & saved_objective_value(unsigned obj_index);
|
inf_eps const & saved_objective_value(unsigned obj_index);
|
||||||
inf_eps current_objective_value(unsigned obj_index);
|
inf_eps current_objective_value(unsigned obj_index);
|
||||||
|
bool objective_is_model_valid(unsigned obj_index) const {
|
||||||
|
return m_valid_objectives[obj_index];
|
||||||
|
}
|
||||||
|
|
||||||
vector<inf_eps> const& get_objective_values();
|
vector<inf_eps> const& get_objective_values();
|
||||||
expr_ref mk_ge(unsigned obj_index, inf_eps const& val);
|
expr_ref mk_ge(unsigned obj_index, inf_eps const& val);
|
||||||
|
@ -124,6 +129,9 @@ namespace opt {
|
||||||
unsigned num_assumptions, expr * const * assumptions,
|
unsigned num_assumptions, expr * const * assumptions,
|
||||||
char const * name = "benchmarks",
|
char const * name = "benchmarks",
|
||||||
char const * logic = "", char const * status = "unknown", char const * attributes = "");
|
char const * logic = "", char const * status = "unknown", char const * attributes = "");
|
||||||
|
|
||||||
|
private:
|
||||||
|
void decrement_value(unsigned i, inf_eps& val);
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -222,7 +222,6 @@ namespace opt {
|
||||||
for (unsigned i = 0; i < m_lower.size() && !m_cancel; ++i) {
|
for (unsigned i = 0; i < m_lower.size() && !m_cancel; ++i) {
|
||||||
if (m_lower[i] < m_upper[i]) {
|
if (m_lower[i] < m_upper[i]) {
|
||||||
mid.push_back((m_upper[i]+m_lower[i])/rational(2));
|
mid.push_back((m_upper[i]+m_lower[i])/rational(2));
|
||||||
//mid.push_back(m_upper[i]);
|
|
||||||
bound = m_s->mk_ge(i, mid[i]);
|
bound = m_s->mk_ge(i, mid[i]);
|
||||||
bounds.push_back(bound);
|
bounds.push_back(bound);
|
||||||
}
|
}
|
||||||
|
@ -373,6 +372,10 @@ namespace opt {
|
||||||
return m_lower[i];
|
return m_lower[i];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool optsmt::objective_is_model_valid(unsigned index) const {
|
||||||
|
return m_s->objective_is_model_valid(index);
|
||||||
|
}
|
||||||
|
|
||||||
inf_eps optsmt::get_upper(unsigned i) const {
|
inf_eps optsmt::get_upper(unsigned i) const {
|
||||||
if (i >= m_upper.size()) return inf_eps();
|
if (i >= m_upper.size()) return inf_eps();
|
||||||
return m_upper[i];
|
return m_upper[i];
|
||||||
|
|
|
@ -58,6 +58,7 @@ namespace opt {
|
||||||
void commit_assignment(unsigned index);
|
void commit_assignment(unsigned index);
|
||||||
inf_eps get_lower(unsigned index) const;
|
inf_eps get_lower(unsigned index) const;
|
||||||
inf_eps get_upper(unsigned index) const;
|
inf_eps get_upper(unsigned index) const;
|
||||||
|
bool objective_is_model_valid(unsigned index) const;
|
||||||
void get_model(model_ref& mdl);
|
void get_model(model_ref& mdl);
|
||||||
|
|
||||||
void update_lower(unsigned idx, inf_eps const& r);
|
void update_lower(unsigned idx, inf_eps const& r);
|
||||||
|
|
|
@ -66,7 +66,7 @@ void display_usage() {
|
||||||
#ifdef Z3GITHASH
|
#ifdef Z3GITHASH
|
||||||
std::cout << " - build hashcode " << STRINGIZE_VALUE_OF(Z3GITHASH);
|
std::cout << " - build hashcode " << STRINGIZE_VALUE_OF(Z3GITHASH);
|
||||||
#endif
|
#endif
|
||||||
std::cout << "]. (C) Copyright 2006-2013 Microsoft Corp.\n";
|
std::cout << "]. (C) Copyright 2006-2014 Microsoft Corp.\n";
|
||||||
std::cout << "Usage: z3 [options] [-file:]file\n";
|
std::cout << "Usage: z3 [options] [-file:]file\n";
|
||||||
std::cout << "\nInput format:\n";
|
std::cout << "\nInput format:\n";
|
||||||
std::cout << " -smt use parser for SMT input format.\n";
|
std::cout << " -smt use parser for SMT input format.\n";
|
||||||
|
|
|
@ -3939,9 +3939,16 @@ namespace smt {
|
||||||
return th->get_value(n, value);
|
return th->get_value(n, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
void context::update_model() {
|
bool context::update_model(bool refinalize) {
|
||||||
mk_proto_model(l_true);
|
final_check_status fcs = FC_DONE;
|
||||||
m_model = m_proto_model->mk_model();
|
if (refinalize) {
|
||||||
|
fcs = final_check();
|
||||||
|
}
|
||||||
|
if (fcs == FC_DONE) {
|
||||||
|
mk_proto_model(l_true);
|
||||||
|
m_model = m_proto_model->mk_model();
|
||||||
|
}
|
||||||
|
return fcs == FC_DONE;
|
||||||
}
|
}
|
||||||
|
|
||||||
void context::mk_proto_model(lbool r) {
|
void context::mk_proto_model(lbool r) {
|
||||||
|
|
|
@ -54,7 +54,11 @@ Revision History:
|
||||||
// the case that each context only references a few expressions.
|
// the case that each context only references a few expressions.
|
||||||
// Using a map instead of a vector for the literals can compress space
|
// Using a map instead of a vector for the literals can compress space
|
||||||
// consumption.
|
// consumption.
|
||||||
|
#ifdef SPARSE_MAP
|
||||||
|
#define USE_BOOL_VAR_VECTOR 0
|
||||||
|
#else
|
||||||
#define USE_BOOL_VAR_VECTOR 1
|
#define USE_BOOL_VAR_VECTOR 1
|
||||||
|
#endif
|
||||||
|
|
||||||
namespace smt {
|
namespace smt {
|
||||||
|
|
||||||
|
@ -69,6 +73,7 @@ namespace smt {
|
||||||
std::string last_failure_as_string() const;
|
std::string last_failure_as_string() const;
|
||||||
void set_progress_callback(progress_callback *callback);
|
void set_progress_callback(progress_callback *callback);
|
||||||
|
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
ast_manager & m_manager;
|
ast_manager & m_manager;
|
||||||
smt_params & m_fparams;
|
smt_params & m_fparams;
|
||||||
|
@ -106,7 +111,7 @@ namespace smt {
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
enode * m_true_enode;
|
enode * m_true_enode;
|
||||||
enode * m_false_enode;
|
enode * m_false_enode;
|
||||||
ptr_vector<enode> m_app2enode; // app -> enode
|
app2enode_t m_app2enode; // app -> enode
|
||||||
ptr_vector<enode> m_enodes;
|
ptr_vector<enode> m_enodes;
|
||||||
plugin_manager<theory> m_theories; // mapping from theory_id -> theory
|
plugin_manager<theory> m_theories; // mapping from theory_id -> theory
|
||||||
ptr_vector<theory> m_theory_set; // set of theories for fast traversal
|
ptr_vector<theory> m_theory_set; // set of theories for fast traversal
|
||||||
|
@ -1390,7 +1395,7 @@ namespace smt {
|
||||||
|
|
||||||
void get_model(model_ref & m) const;
|
void get_model(model_ref & m) const;
|
||||||
|
|
||||||
void update_model();
|
bool update_model(bool refinalize);
|
||||||
|
|
||||||
void get_proto_model(proto_model_ref & m) const;
|
void get_proto_model(proto_model_ref & m) const;
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ namespace smt {
|
||||||
/**
|
/**
|
||||||
\brief Initialize an enode in the given memory position.
|
\brief Initialize an enode in the given memory position.
|
||||||
*/
|
*/
|
||||||
enode * enode::init(ast_manager & m, void * mem, ptr_vector<enode> const & app2enode, app * owner,
|
enode * enode::init(ast_manager & m, void * mem, app2enode_t const & app2enode, app * owner,
|
||||||
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
||||||
bool cgc_enabled, bool update_children_parent) {
|
bool cgc_enabled, bool update_children_parent) {
|
||||||
SASSERT(m.is_bool(owner) || !merge_tf);
|
SASSERT(m.is_bool(owner) || !merge_tf);
|
||||||
|
@ -60,7 +60,7 @@ namespace smt {
|
||||||
return n;
|
return n;
|
||||||
}
|
}
|
||||||
|
|
||||||
enode * enode::mk(ast_manager & m, region & r, ptr_vector<enode> const & app2enode, app * owner,
|
enode * enode::mk(ast_manager & m, region & r, app2enode_t const & app2enode, app * owner,
|
||||||
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
||||||
bool cgc_enabled, bool update_children_parent) {
|
bool cgc_enabled, bool update_children_parent) {
|
||||||
SASSERT(m.is_bool(owner) || !merge_tf);
|
SASSERT(m.is_bool(owner) || !merge_tf);
|
||||||
|
@ -69,7 +69,7 @@ namespace smt {
|
||||||
return init(m, mem, app2enode, owner, generation, suppress_args, merge_tf, iscope_lvl, cgc_enabled, update_children_parent);
|
return init(m, mem, app2enode, owner, generation, suppress_args, merge_tf, iscope_lvl, cgc_enabled, update_children_parent);
|
||||||
}
|
}
|
||||||
|
|
||||||
enode * enode::mk_dummy(ast_manager & m, ptr_vector<enode> const & app2enode, app * owner) {
|
enode * enode::mk_dummy(ast_manager & m, app2enode_t const & app2enode, app * owner) {
|
||||||
unsigned sz = get_enode_size(owner->get_num_args());
|
unsigned sz = get_enode_size(owner->get_num_args());
|
||||||
void * mem = alloc_svect(char, sz);
|
void * mem = alloc_svect(char, sz);
|
||||||
return init(m, mem, app2enode, owner, 0, false, false, 0, true, false);
|
return init(m, mem, app2enode, owner, 0, false, false, 0, true, false);
|
||||||
|
|
|
@ -38,6 +38,29 @@ namespace smt {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/** \ brief Use sparse maps in SMT solver.
|
||||||
|
|
||||||
|
Define this to use hash maps rather than vectors over ast
|
||||||
|
nodes. This is useful in the case there are many solvers, each
|
||||||
|
referencing few nodes from a large ast manager. There is some
|
||||||
|
unknown performance penalty for this. */
|
||||||
|
|
||||||
|
// #define SPARSE_MAP
|
||||||
|
|
||||||
|
#ifndef SPARSE_MAP
|
||||||
|
typedef ptr_vector<enode> app2enode_t; // app -> enode
|
||||||
|
#else
|
||||||
|
class app2enode_t : public u_map<enode *> {
|
||||||
|
public:
|
||||||
|
void setx(unsigned x, enode *val, enode *def){
|
||||||
|
if(val == 0)
|
||||||
|
erase(x);
|
||||||
|
else
|
||||||
|
insert(x,val);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
#endif
|
||||||
|
|
||||||
class tmp_enode;
|
class tmp_enode;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -115,7 +138,7 @@ namespace smt {
|
||||||
|
|
||||||
friend class tmp_enode;
|
friend class tmp_enode;
|
||||||
|
|
||||||
static enode * init(ast_manager & m, void * mem, ptr_vector<enode> const & app2enode, app * owner,
|
static enode * init(ast_manager & m, void * mem, app2enode_t const & app2enode, app * owner,
|
||||||
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
||||||
bool cgc_enabled, bool update_children_parent);
|
bool cgc_enabled, bool update_children_parent);
|
||||||
public:
|
public:
|
||||||
|
@ -124,11 +147,11 @@ namespace smt {
|
||||||
return sizeof(enode) + num_args * sizeof(enode*);
|
return sizeof(enode) + num_args * sizeof(enode*);
|
||||||
}
|
}
|
||||||
|
|
||||||
static enode * mk(ast_manager & m, region & r, ptr_vector<enode> const & app2enode, app * owner,
|
static enode * mk(ast_manager & m, region & r, app2enode_t const & app2enode, app * owner,
|
||||||
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
unsigned generation, bool suppress_args, bool merge_tf, unsigned iscope_lvl,
|
||||||
bool cgc_enabled, bool update_children_parent);
|
bool cgc_enabled, bool update_children_parent);
|
||||||
|
|
||||||
static enode * mk_dummy(ast_manager & m, ptr_vector<enode> const & app2enode, app * owner);
|
static enode * mk_dummy(ast_manager & m, app2enode_t const & app2enode, app * owner);
|
||||||
|
|
||||||
static void del_dummy(enode * n) { dealloc_svect(reinterpret_cast<char*>(n)); }
|
static void del_dummy(enode * n) { dealloc_svect(reinterpret_cast<char*>(n)); }
|
||||||
|
|
||||||
|
|
|
@ -874,13 +874,13 @@ namespace smt {
|
||||||
|
|
||||||
void add_tmp_row(row & r1, numeral const & coeff, row const & r2);
|
void add_tmp_row(row & r1, numeral const & coeff, row const & r2);
|
||||||
theory_var pick_var_to_leave(bool has_int, theory_var x_j, bool inc, numeral & a_ij, inf_numeral & gain, bool& skiped_row);
|
theory_var pick_var_to_leave(bool has_int, theory_var x_j, bool inc, numeral & a_ij, inf_numeral & gain, bool& skiped_row);
|
||||||
bool is_safe_to_leave(theory_var x, bool& has_int);
|
bool is_safe_to_leave(theory_var x, bool& has_int, bool& is_shared);
|
||||||
bool move_to_bound(theory_var x_i, bool inc);
|
bool move_to_bound(theory_var x_i, bool inc);
|
||||||
template<bool invert>
|
template<bool invert>
|
||||||
void add_tmp_row_entry(row & r, numeral const & coeff, theory_var v);
|
void add_tmp_row_entry(row & r, numeral const & coeff, theory_var v);
|
||||||
enum max_min_t { UNBOUNDED, AT_BOUND, OPTIMIZED, BEST_EFFORT};
|
enum max_min_t { UNBOUNDED, AT_BOUND, OPTIMIZED, BEST_EFFORT};
|
||||||
max_min_t max_min(theory_var v, bool max);
|
max_min_t max_min(theory_var v, bool max, bool& has_shared);
|
||||||
max_min_t max_min(row & r, bool max);
|
max_min_t max_min(row & r, bool max, bool& has_shared);
|
||||||
bool max_min(svector<theory_var> const & vars);
|
bool max_min(svector<theory_var> const & vars);
|
||||||
|
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
|
@ -1016,7 +1016,7 @@ namespace smt {
|
||||||
// Optimization
|
// Optimization
|
||||||
//
|
//
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
virtual inf_eps_rational<inf_rational> maximize(theory_var v, expr_ref& blocker);
|
virtual inf_eps_rational<inf_rational> maximize(theory_var v, expr_ref& blocker, bool& has_shared);
|
||||||
virtual inf_eps_rational<inf_rational> value(theory_var v);
|
virtual inf_eps_rational<inf_rational> value(theory_var v);
|
||||||
virtual theory_var add_objective(app* term);
|
virtual theory_var add_objective(app* term);
|
||||||
virtual expr* mk_ge(filter_model_converter& fm, theory_var v, inf_numeral const& val);
|
virtual expr* mk_ge(filter_model_converter& fm, theory_var v, inf_numeral const& val);
|
||||||
|
|
|
@ -928,11 +928,10 @@ namespace smt {
|
||||||
|
|
||||||
*/
|
*/
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
bool theory_arith<Ext>::is_safe_to_leave(theory_var x, bool& has_int) {
|
bool theory_arith<Ext>::is_safe_to_leave(theory_var x, bool& has_int, bool& shared) {
|
||||||
|
|
||||||
if (get_context().is_shared(get_enode(x))) {
|
context& ctx = get_context();
|
||||||
return false;
|
shared |= ctx.is_shared(get_enode(x));
|
||||||
}
|
|
||||||
column & c = m_columns[x];
|
column & c = m_columns[x];
|
||||||
typename svector<col_entry>::iterator it = c.begin_entries();
|
typename svector<col_entry>::iterator it = c.begin_entries();
|
||||||
typename svector<col_entry>::iterator end = c.end_entries();
|
typename svector<col_entry>::iterator end = c.end_entries();
|
||||||
|
@ -944,8 +943,9 @@ namespace smt {
|
||||||
numeral const & coeff = r[it->m_row_idx].m_coeff;
|
numeral const & coeff = r[it->m_row_idx].m_coeff;
|
||||||
if (s != null_theory_var && is_int(s)) has_int = true;
|
if (s != null_theory_var && is_int(s)) has_int = true;
|
||||||
bool is_unsafe = (s != null_theory_var && is_int(s) && !coeff.is_int());
|
bool is_unsafe = (s != null_theory_var && is_int(s) && !coeff.is_int());
|
||||||
is_unsafe = is_unsafe || (s != null_theory_var && get_context().is_shared(get_enode(s)));
|
shared |= (s != null_theory_var && ctx.is_shared(get_enode(s)));
|
||||||
TRACE("opt", tout << "is v" << x << " safe to leave for v" << s << "? " << (is_unsafe?"no":"yes") << " " << (has_int?"int":"real") << "\n";
|
TRACE("opt", tout << "is v" << x << " safe to leave for v" << s
|
||||||
|
<< "? " << (is_unsafe?"no":"yes") << " " << (has_int?"int":"real") << "\n";
|
||||||
display_row(tout, r, true););
|
display_row(tout, r, true););
|
||||||
if (is_unsafe) return false;
|
if (is_unsafe) return false;
|
||||||
}
|
}
|
||||||
|
@ -1072,10 +1072,12 @@ namespace smt {
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
inf_eps_rational<inf_rational> theory_arith<Ext>::maximize(theory_var v, expr_ref& blocker) {
|
inf_eps_rational<inf_rational> theory_arith<Ext>::maximize(theory_var v, expr_ref& blocker, bool& has_shared) {
|
||||||
TRACE("bound_bug", display_var(tout, v); display(tout););
|
TRACE("bound_bug", display_var(tout, v); display(tout););
|
||||||
max_min_t r = max_min(v, true);
|
has_shared = false;
|
||||||
|
max_min_t r = max_min(v, true, has_shared);
|
||||||
if (r == UNBOUNDED) {
|
if (r == UNBOUNDED) {
|
||||||
|
has_shared = false;
|
||||||
blocker = get_manager().mk_false();
|
blocker = get_manager().mk_false();
|
||||||
return inf_eps_rational<inf_rational>::infinity();
|
return inf_eps_rational<inf_rational>::infinity();
|
||||||
}
|
}
|
||||||
|
@ -1299,10 +1301,11 @@ namespace smt {
|
||||||
Return true if succeeded.
|
Return true if succeeded.
|
||||||
*/
|
*/
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
typename theory_arith<Ext>::max_min_t theory_arith<Ext>::max_min(row & r, bool max) {
|
typename theory_arith<Ext>::max_min_t theory_arith<Ext>::max_min(row & r, bool max, bool& has_shared) {
|
||||||
TRACE("max_min", tout << "max_min...\n";);
|
TRACE("max_min", tout << "max_min...\n";);
|
||||||
m_stats.m_max_min++;
|
m_stats.m_max_min++;
|
||||||
bool skipped_row = false;
|
bool skipped_row = false;
|
||||||
|
has_shared = false;
|
||||||
|
|
||||||
SASSERT(valid_row_assignment());
|
SASSERT(valid_row_assignment());
|
||||||
SASSERT(satisfy_bounds());
|
SASSERT(satisfy_bounds());
|
||||||
|
@ -1332,7 +1335,7 @@ namespace smt {
|
||||||
bool has_int = false;
|
bool has_int = false;
|
||||||
if ((curr_inc && at_upper(curr_x_j)) || (!curr_inc && at_lower(curr_x_j)))
|
if ((curr_inc && at_upper(curr_x_j)) || (!curr_inc && at_lower(curr_x_j)))
|
||||||
continue; // variable cannot be used for max/min.
|
continue; // variable cannot be used for max/min.
|
||||||
if (!is_safe_to_leave(curr_x_j, has_int)) {
|
if (!is_safe_to_leave(curr_x_j, has_int, has_shared)) {
|
||||||
skipped_row = true;
|
skipped_row = true;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
@ -1536,7 +1539,7 @@ namespace smt {
|
||||||
\brief Maximize/Minimize the given variable. The bounds of v are update if procedure succeeds.
|
\brief Maximize/Minimize the given variable. The bounds of v are update if procedure succeeds.
|
||||||
*/
|
*/
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
typename theory_arith<Ext>::max_min_t theory_arith<Ext>::max_min(theory_var v, bool max) {
|
typename theory_arith<Ext>::max_min_t theory_arith<Ext>::max_min(theory_var v, bool max, bool& has_shared) {
|
||||||
expr* e = get_enode(v)->get_owner();
|
expr* e = get_enode(v)->get_owner();
|
||||||
SASSERT(valid_row_assignment());
|
SASSERT(valid_row_assignment());
|
||||||
SASSERT(satisfy_bounds());
|
SASSERT(satisfy_bounds());
|
||||||
|
@ -1558,7 +1561,7 @@ namespace smt {
|
||||||
add_tmp_row_entry<true>(m_tmp_row, it->m_coeff, it->m_var);
|
add_tmp_row_entry<true>(m_tmp_row, it->m_coeff, it->m_var);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
max_min_t r = max_min(m_tmp_row, max);
|
max_min_t r = max_min(m_tmp_row, max, has_shared);
|
||||||
if (r == OPTIMIZED) {
|
if (r == OPTIMIZED) {
|
||||||
TRACE("opt", tout << mk_pp(e, get_manager()) << " " << (max ? "max" : "min") << " value is: " << get_value(v) << "\n";
|
TRACE("opt", tout << mk_pp(e, get_manager()) << " " << (max ? "max" : "min") << " value is: " << get_value(v) << "\n";
|
||||||
display_row(tout, m_tmp_row, true); display_row_info(tout, m_tmp_row););
|
display_row(tout, m_tmp_row, true); display_row_info(tout, m_tmp_row););
|
||||||
|
@ -1581,12 +1584,13 @@ namespace smt {
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
bool theory_arith<Ext>::max_min(svector<theory_var> const & vars) {
|
bool theory_arith<Ext>::max_min(svector<theory_var> const & vars) {
|
||||||
bool succ = false;
|
bool succ = false;
|
||||||
|
bool has_shared = false;
|
||||||
svector<theory_var>::const_iterator it = vars.begin();
|
svector<theory_var>::const_iterator it = vars.begin();
|
||||||
svector<theory_var>::const_iterator end = vars.end();
|
svector<theory_var>::const_iterator end = vars.end();
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
if (max_min(*it, true) == OPTIMIZED)
|
if (max_min(*it, true, has_shared) == OPTIMIZED && !has_shared)
|
||||||
succ = true;
|
succ = true;
|
||||||
if (max_min(*it, false) == OPTIMIZED)
|
if (max_min(*it, false, has_shared) == OPTIMIZED && !has_shared)
|
||||||
succ = true;
|
succ = true;
|
||||||
}
|
}
|
||||||
if (succ) {
|
if (succ) {
|
||||||
|
|
|
@ -376,8 +376,10 @@ namespace smt {
|
||||||
enode_vector::const_iterator end = r->end_parents();
|
enode_vector::const_iterator end = r->end_parents();
|
||||||
for (; it != end; ++it) {
|
for (; it != end; ++it) {
|
||||||
enode * parent = *it;
|
enode * parent = *it;
|
||||||
|
#if 0
|
||||||
if (!ctx.is_relevant(parent))
|
if (!ctx.is_relevant(parent))
|
||||||
continue;
|
continue;
|
||||||
|
#endif
|
||||||
unsigned num_args = parent->get_num_args();
|
unsigned num_args = parent->get_num_args();
|
||||||
if (is_store(parent)) {
|
if (is_store(parent)) {
|
||||||
SET_ARRAY(parent->get_arg(0));
|
SET_ARRAY(parent->get_arg(0));
|
||||||
|
@ -399,6 +401,7 @@ namespace smt {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#if 0
|
||||||
void theory_array_base::collect_shared_vars(sbuffer<theory_var> & result) {
|
void theory_array_base::collect_shared_vars(sbuffer<theory_var> & result) {
|
||||||
TRACE("array_shared", tout << "collecting shared vars...\n";);
|
TRACE("array_shared", tout << "collecting shared vars...\n";);
|
||||||
context & ctx = get_context();
|
context & ctx = get_context();
|
||||||
|
@ -420,6 +423,31 @@ namespace smt {
|
||||||
}
|
}
|
||||||
unmark_enodes(to_unmark.size(), to_unmark.c_ptr());
|
unmark_enodes(to_unmark.size(), to_unmark.c_ptr());
|
||||||
}
|
}
|
||||||
|
#else
|
||||||
|
void theory_array_base::collect_shared_vars(sbuffer<theory_var> & result) {
|
||||||
|
TRACE("array_shared", tout << "collecting shared vars...\n";);
|
||||||
|
context & ctx = get_context();
|
||||||
|
ptr_buffer<enode> to_unmark;
|
||||||
|
unsigned num_vars = get_num_vars();
|
||||||
|
for (unsigned i = 0; i < num_vars; i++) {
|
||||||
|
enode * n = get_enode(i);
|
||||||
|
if (ctx.is_relevant(n)) {
|
||||||
|
enode * r = n->get_root();
|
||||||
|
if (!r->is_marked()){
|
||||||
|
if(is_array_sort(r) && ctx.is_shared(r)) {
|
||||||
|
TRACE("array_shared", tout << "new shared var: #" << r->get_owner_id() << "\n";);
|
||||||
|
theory_var r_th_var = r->get_th_var(get_id());
|
||||||
|
SASSERT(r_th_var != null_theory_var);
|
||||||
|
result.push_back(r_th_var);
|
||||||
|
}
|
||||||
|
r->set_mark();
|
||||||
|
to_unmark.push_back(r);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
unmark_enodes(to_unmark.size(), to_unmark.c_ptr());
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
/**
|
/**
|
||||||
\brief Create interface variables for shared array variables.
|
\brief Create interface variables for shared array variables.
|
||||||
|
|
|
@ -266,7 +266,7 @@ namespace smt {
|
||||||
//
|
//
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
|
|
||||||
virtual inf_eps_rational<inf_rational> maximize(theory_var v, expr_ref& blocker);
|
virtual inf_eps_rational<inf_rational> maximize(theory_var v, expr_ref& blocker, bool& has_shared);
|
||||||
virtual inf_eps_rational<inf_rational> value(theory_var v);
|
virtual inf_eps_rational<inf_rational> value(theory_var v);
|
||||||
virtual theory_var add_objective(app* term);
|
virtual theory_var add_objective(app* term);
|
||||||
virtual expr_ref mk_gt(theory_var v, inf_rational const& val);
|
virtual expr_ref mk_gt(theory_var v, inf_rational const& val);
|
||||||
|
|
|
@ -889,11 +889,12 @@ namespace smt {
|
||||||
}
|
}
|
||||||
|
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
inf_eps_rational<inf_rational> theory_dense_diff_logic<Ext>::maximize(theory_var v, expr_ref& blocker) {
|
inf_eps_rational<inf_rational> theory_dense_diff_logic<Ext>::maximize(theory_var v, expr_ref& blocker, bool& has_shared) {
|
||||||
typedef simplex::simplex<simplex::mpq_ext> Simplex;
|
typedef simplex::simplex<simplex::mpq_ext> Simplex;
|
||||||
Simplex S;
|
Simplex S;
|
||||||
ast_manager& m = get_manager();
|
ast_manager& m = get_manager();
|
||||||
objective_term const& objective = m_objectives[v];
|
objective_term const& objective = m_objectives[v];
|
||||||
|
has_shared = false;
|
||||||
|
|
||||||
IF_VERBOSE(1,
|
IF_VERBOSE(1,
|
||||||
for (unsigned i = 0; i < objective.size(); ++i) {
|
for (unsigned i = 0; i < objective.size(); ++i) {
|
||||||
|
|
|
@ -320,7 +320,7 @@ namespace smt {
|
||||||
//
|
//
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
|
|
||||||
virtual inf_eps maximize(theory_var v, expr_ref& blocker);
|
virtual inf_eps maximize(theory_var v, expr_ref& blocker, bool& has_shared);
|
||||||
virtual inf_eps value(theory_var v);
|
virtual inf_eps value(theory_var v);
|
||||||
virtual theory_var add_objective(app* term);
|
virtual theory_var add_objective(app* term);
|
||||||
virtual expr_ref mk_gt(theory_var v, inf_rational const& val);
|
virtual expr_ref mk_gt(theory_var v, inf_rational const& val);
|
||||||
|
|
|
@ -1187,8 +1187,9 @@ typename theory_diff_logic<Ext>::inf_eps theory_diff_logic<Ext>::value(theory_va
|
||||||
|
|
||||||
template<typename Ext>
|
template<typename Ext>
|
||||||
typename theory_diff_logic<Ext>::inf_eps
|
typename theory_diff_logic<Ext>::inf_eps
|
||||||
theory_diff_logic<Ext>::maximize(theory_var v, expr_ref& blocker) {
|
theory_diff_logic<Ext>::maximize(theory_var v, expr_ref& blocker, bool& has_shared) {
|
||||||
|
|
||||||
|
has_shared = false;
|
||||||
Simplex& S = m_S;
|
Simplex& S = m_S;
|
||||||
ast_manager& m = get_manager();
|
ast_manager& m = get_manager();
|
||||||
|
|
||||||
|
|
|
@ -31,7 +31,7 @@ namespace smt {
|
||||||
public:
|
public:
|
||||||
typedef inf_eps_rational<inf_rational> inf_eps;
|
typedef inf_eps_rational<inf_rational> inf_eps;
|
||||||
virtual inf_eps value(theory_var) = 0;
|
virtual inf_eps value(theory_var) = 0;
|
||||||
virtual inf_eps maximize(theory_var v, expr_ref& blocker) = 0;
|
virtual inf_eps maximize(theory_var v, expr_ref& blocker, bool& has_shared) = 0;
|
||||||
virtual theory_var add_objective(app* term) = 0;
|
virtual theory_var add_objective(app* term) = 0;
|
||||||
virtual expr* mk_ge(filter_model_converter& fm, theory_var v, inf_eps const& val) { UNREACHABLE(); return 0; }
|
virtual expr* mk_ge(filter_model_converter& fm, theory_var v, inf_eps const& val) { UNREACHABLE(); return 0; }
|
||||||
bool is_linear(ast_manager& m, expr* term);
|
bool is_linear(ast_manager& m, expr* term);
|
||||||
|
|
|
@ -111,14 +111,25 @@ void fpa2bv_model_converter::convert(model * bv_mdl, model * float_mdl) {
|
||||||
unsigned ebits = fu.get_ebits(var->get_range());
|
unsigned ebits = fu.get_ebits(var->get_range());
|
||||||
unsigned sbits = fu.get_sbits(var->get_range());
|
unsigned sbits = fu.get_sbits(var->get_range());
|
||||||
|
|
||||||
expr_ref sgn(m), sig(m), exp(m);
|
expr_ref sgn(m), sig(m), exp(m);
|
||||||
sgn = bv_mdl->get_const_interp(to_app(a->get_arg(0))->get_decl());
|
bv_mdl->eval(a->get_arg(0), sgn, true);
|
||||||
sig = bv_mdl->get_const_interp(to_app(a->get_arg(1))->get_decl());
|
bv_mdl->eval(a->get_arg(1), sig, true);
|
||||||
exp = bv_mdl->get_const_interp(to_app(a->get_arg(2))->get_decl());
|
bv_mdl->eval(a->get_arg(2), exp, true);
|
||||||
|
|
||||||
|
SASSERT(a->is_app_of(fu.get_family_id(), OP_TO_FLOAT));
|
||||||
|
|
||||||
|
#ifdef Z3DEBUG
|
||||||
|
SASSERT(to_app(a->get_arg(0))->get_decl()->get_arity() == 0);
|
||||||
|
SASSERT(to_app(a->get_arg(1))->get_decl()->get_arity() == 0);
|
||||||
|
SASSERT(to_app(a->get_arg(1))->get_decl()->get_arity() == 0);
|
||||||
seen.insert(to_app(a->get_arg(0))->get_decl());
|
seen.insert(to_app(a->get_arg(0))->get_decl());
|
||||||
seen.insert(to_app(a->get_arg(1))->get_decl());
|
seen.insert(to_app(a->get_arg(1))->get_decl());
|
||||||
seen.insert(to_app(a->get_arg(2))->get_decl());
|
seen.insert(to_app(a->get_arg(2))->get_decl());
|
||||||
|
#else
|
||||||
|
SASSERT(a->get_arg(0)->get_kind() == OP_EXTRACT);
|
||||||
|
SASSERT(to_app(a->get_arg(0))->get_arg(0)->get_kind() == OP_EXTRACT);
|
||||||
|
seen.insert(to_app(to_app(a->get_arg(0))->get_arg(0))->get_decl());
|
||||||
|
#endif
|
||||||
|
|
||||||
if (!sgn && !sig && !exp)
|
if (!sgn && !sig && !exp)
|
||||||
continue;
|
continue;
|
||||||
|
|
|
@ -131,7 +131,7 @@ public:
|
||||||
value const& get(key const& k, value const& default_value) const {
|
value const& get(key const& k, value const& default_value) const {
|
||||||
entry* e = find_core(k);
|
entry* e = find_core(k);
|
||||||
if (e) {
|
if (e) {
|
||||||
return e->m_value;
|
return e->get_data().m_value;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
return default_value;
|
return default_value;
|
||||||
|
|
|
@ -1535,7 +1535,7 @@ bool mpz_manager<SYNCH>::is_power_of_two(mpz const & a, unsigned & shift) {
|
||||||
return false;
|
return false;
|
||||||
if (is_small(a)) {
|
if (is_small(a)) {
|
||||||
if (::is_power_of_two(a.m_val)) {
|
if (::is_power_of_two(a.m_val)) {
|
||||||
shift = ::log2(a.m_val);
|
shift = ::log2((unsigned)a.m_val);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
|
@ -1838,7 +1838,7 @@ unsigned mpz_manager<SYNCH>::log2(mpz const & a) {
|
||||||
if (is_nonpos(a))
|
if (is_nonpos(a))
|
||||||
return 0;
|
return 0;
|
||||||
if (is_small(a))
|
if (is_small(a))
|
||||||
return ::log2(a.m_val);
|
return ::log2((unsigned)a.m_val);
|
||||||
#ifndef _MP_GMP
|
#ifndef _MP_GMP
|
||||||
COMPILE_TIME_ASSERT(sizeof(digit_t) == 8 || sizeof(digit_t) == 4);
|
COMPILE_TIME_ASSERT(sizeof(digit_t) == 8 || sizeof(digit_t) == 4);
|
||||||
mpz_cell * c = a.m_ptr;
|
mpz_cell * c = a.m_ptr;
|
||||||
|
@ -1860,7 +1860,7 @@ unsigned mpz_manager<SYNCH>::mlog2(mpz const & a) {
|
||||||
if (is_nonneg(a))
|
if (is_nonneg(a))
|
||||||
return 0;
|
return 0;
|
||||||
if (is_small(a))
|
if (is_small(a))
|
||||||
return ::log2(-a.m_val);
|
return ::log2((unsigned)-a.m_val);
|
||||||
#ifndef _MP_GMP
|
#ifndef _MP_GMP
|
||||||
COMPILE_TIME_ASSERT(sizeof(digit_t) == 8 || sizeof(digit_t) == 4);
|
COMPILE_TIME_ASSERT(sizeof(digit_t) == 8 || sizeof(digit_t) == 4);
|
||||||
mpz_cell * c = a.m_ptr;
|
mpz_cell * c = a.m_ptr;
|
||||||
|
|
Loading…
Reference in a new issue