diff --git a/.classpath b/.classpath index 2249abb932..d887208179 100644 --- a/.classpath +++ b/.classpath @@ -21,11 +21,12 @@ - + + diff --git a/.gitignore b/.gitignore index 46f4c466ab..35f1d9ae58 100644 --- a/.gitignore +++ b/.gitignore @@ -1,3 +1,8 @@ +/native-library-files/stp_project/stp +/native-library-files/stp_project/stpJ/build +/native-library-files/stp_project/stpJ/files +/native-library-files/stp_project/stpJ/install_include + .idea/workspace.xml .factorypath diff --git a/.settings/org.eclipse.jdt.core.prefs b/.settings/org.eclipse.jdt.core.prefs index 52e6e37189..5611fc4f24 100644 --- a/.settings/org.eclipse.jdt.core.prefs +++ b/.settings/org.eclipse.jdt.core.prefs @@ -133,6 +133,7 @@ org.eclipse.jdt.core.compiler.problem.unusedTypeParameter=warning org.eclipse.jdt.core.compiler.problem.unusedWarningToken=ignore org.eclipse.jdt.core.compiler.problem.varargsArgumentNeedCast=warning org.eclipse.jdt.core.compiler.processAnnotations=enabled +org.eclipse.jdt.core.compiler.release=disabled org.eclipse.jdt.core.compiler.source=1.8 org.eclipse.jdt.core.formatter.align_fields_grouping_blank_lines=2147483647 org.eclipse.jdt.core.formatter.align_type_members_on_columns=false @@ -223,11 +224,9 @@ org.eclipse.jdt.core.formatter.indent_statements_compare_to_body=true org.eclipse.jdt.core.formatter.indent_switchstatements_compare_to_cases=true org.eclipse.jdt.core.formatter.indent_switchstatements_compare_to_switch=true org.eclipse.jdt.core.formatter.indentation.size=4 -org.eclipse.jdt.core.formatter.insert_new_line_after_annotation=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_enum_constant=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_field=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_local_variable=insert -org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_member=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_method=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_package=insert org.eclipse.jdt.core.formatter.insert_new_line_after_annotation_on_parameter=do not insert diff --git a/build.xml b/build.xml index c411fd59f6..073e5e33da 100644 --- a/build.xml +++ b/build.xml @@ -39,6 +39,9 @@ + + + diff --git a/lib/native/x86_64-linux/libstpJapi.so b/lib/native/x86_64-linux/libstpJapi.so new file mode 100755 index 0000000000..b645bb7638 Binary files /dev/null and b/lib/native/x86_64-linux/libstpJapi.so differ diff --git a/native-library-files/stp_project/.vscode/c_cpp_properties.json b/native-library-files/stp_project/.vscode/c_cpp_properties.json new file mode 100644 index 0000000000..31fe2a8444 --- /dev/null +++ b/native-library-files/stp_project/.vscode/c_cpp_properties.json @@ -0,0 +1,17 @@ +{ + "configurations": [ + { + "name": "Linux", + "includePath": [ + "${workspaceFolder}/**" + ], + "defines": [], + "compilerPath": "/usr/bin/gcc", + "cStandard": "c11", + "cppStandard": "c++17", + "intelliSenseMode": "clang-x64", + "compileCommands": "${workspaceFolder}/stp/build/compile_commands.json" + } + ], + "version": 4 +} \ No newline at end of file diff --git a/native-library-files/stp_project/.vscode/settings.json b/native-library-files/stp_project/.vscode/settings.json new file mode 100644 index 0000000000..b54e2d264c --- /dev/null +++ b/native-library-files/stp_project/.vscode/settings.json @@ -0,0 +1,55 @@ +{ + "files.associations": { + "cstdlib": "cpp", + "extstpswigapi.h": "c", + "cassert": "cpp", + "iostream": "cpp", + "cctype": "cpp", + "clocale": "cpp", + "cmath": "cpp", + "cstdarg": "cpp", + "cstddef": "cpp", + "cstdio": "cpp", + "cstring": "cpp", + "ctime": "cpp", + "cwchar": "cpp", + "cwctype": "cpp", + "array": "cpp", + "atomic": "cpp", + "*.tcc": "cpp", + "cstdint": "cpp", + "deque": "cpp", + "list": "cpp", + "unordered_map": "cpp", + "unordered_set": "cpp", + "vector": "cpp", + "exception": "cpp", + "algorithm": "cpp", + "functional": "cpp", + "iterator": "cpp", + "map": "cpp", + "memory": "cpp", + "memory_resource": "cpp", + "numeric": "cpp", + "random": "cpp", + "set": "cpp", + "string": "cpp", + "system_error": "cpp", + "tuple": "cpp", + "type_traits": "cpp", + "utility": "cpp", + "fstream": "cpp", + "initializer_list": "cpp", + "iomanip": "cpp", + "iosfwd": "cpp", + "istream": "cpp", + "limits": "cpp", + "new": "cpp", + "ostream": "cpp", + "sstream": "cpp", + "stdexcept": "cpp", + "streambuf": "cpp", + "cinttypes": "cpp", + "typeinfo": "cpp" + } +} \ No newline at end of file diff --git a/native-library-files/stp_project/Readme.md b/native-library-files/stp_project/Readme.md new file mode 100644 index 0000000000..6f3055121d --- /dev/null +++ b/native-library-files/stp_project/Readme.md @@ -0,0 +1,41 @@ + +# README: Scripts and Directory Structure + +## ./stp + +This contains STP source code, cloned from its Github repo. This folder is created by a script that clones the repo. + +## ./stp_extend + +In order to extend the STP api, it was necessary to directly modify some source files. This folder contains these source files : *c_interface.h*, *ext_c_interface.cpp* and *CMakeLists.txt*. These files should be modified directly in these folder since the build script will be copying them to replace the one in STP source. **Please edit the source for extending STP inside this folder** + +## ./stpJ + +Contains the folders and files necessary to build the Java Binding for STP. There is an attempt to extend the interface at the level of Java Binding the sources for this is in folders */stpJ/include/* and */stpJ/src/*. + +The raw STP library (before binding) is installed inside the folder */stpJ/install_include/*, Cmake should locate it there. + +*/stpJ/build/* contains build files for the Java binding. The resulting object files are copied to their appropriate java-smt's library folders. + +*/stpJ/StpJavaApi.i* is the SWIG interface used to generate an interface for the Java Binding. + +## ./install_prereq.sh and ./dependencies + +STP requires some dependencies, *./install_prereq.sh* calls a script inside *./dependencies/* which installs them. It also clones cryptominisat into this folder and build it before installing it. + +## ./clean_up.sh + +This removes build files for the binding and also the source files for cryptominisat. + +## ./clean_and_build_stp.sh + +This removes build files for the Java binding, then copies the STP extension files to replace corresponding files in the STP source. It then builds the raw STP. + +## ./clean_clone_build_stp.sh + +This removes all build files for raw STP and the Java binding. It also removes the STP source and reclones it for it Github repo. Then copies the STP extension files to replace corresponding files in the STP source. It then builds the raw STP. + +## ./build_raw_to_java_api.sh + +This is the script that builds the Java Binding for STP. It clones the STP source from Githib if not found otherwise it asks if you want to rebuild the raw STP, if 'y' it calls */clean_and_build_stp.sh* before going ahead to build the Java Binding. + diff --git a/native-library-files/stp_project/build_raw_to_java_api.sh b/native-library-files/stp_project/build_raw_to_java_api.sh new file mode 100644 index 0000000000..2226ebb755 --- /dev/null +++ b/native-library-files/stp_project/build_raw_to_java_api.sh @@ -0,0 +1,76 @@ +#!/usr/bin/etc/ bash + +set -e +set -u + +# Gets the normal STP library, if not found it is cloned and built. +# Afterwards, using SWIG we define a new API and generate its JNI bindings +# The new API is rebuild as a shared library and the JNI Java classes generated +# The resulting .so library and the binding classes (compile to a .jar) +# are copied into appropriate folder in java-smt. + + +STP_LIB=./stp/build/lib/libstp.so + + +PRJ_DIR="${PWD%/*/*}" +PRJ_NAME=$(basename "$PRJ_DIR") + + +# confirm expected directory structure +# MUST BE like java-smt/../../this_script.sh +if [[ "$PRJ_NAME" != "java-smt" ]] + then + echo "this script is not place in the proper directory" >&2 + echo "this script expects to reside in somewhere like\ + .../java-smt/dir_1/dir_2/build_raw_to_java_api.sh" >&2 + exit 1 +fi + +# set destination directory for final libraries +JAR_LIB_DIR="${PRJ_DIR}/lib/" +SO_LIB_DIR="${JAR_LIB_DIR}/native/x86_64-linux/" + + +# echo $FILE +echo --- +if [! -f "$STP_LIB" ] + then + echo "I can't find the STP library file. I am making a new one ..." + sh clean_clone_build_stp.sh +else + echo -n "To rebuild stp press 'y' " + read re_bld + if [ "$re_bld" != "${re_bld#[Yy]}" ] ;then + sh clean_and_build_stp.sh + fi + +fi + +cd ./stpJ +[ ! -f ./build ] && mkdir ./build || rm -rf ./build/* ||: + +cd ./build + +echo "Now building the JAVA API for STP ..." + +cmake .. #-DCMAKE_BUILD_TYPE=Debug +cmake --build . + +echo +echo "BUILD SUCCESSFUL" +echo + +ls + +# copy API - jar and - so into javasmt +echo +echo "copying library files into JavaSMT (old files are overwritten) ... ..." + +# cp ./stpJavaAPI.jar /home/lubuntu/SAHEED/gsoc/CODE/java-smt/lib/ +# cp ./libstpJapi.so /home/lubuntu/SAHEED/gsoc/CODE/java-smt/lib/native/x86_64-linux/ + +cp ./stpJavaAPI.jar $JAR_LIB_DIR +cp ./libstpJapi.so $SO_LIB_DIR + +echo "SUCCESS" \ No newline at end of file diff --git a/native-library-files/stp_project/clean_and_build_stp.sh b/native-library-files/stp_project/clean_and_build_stp.sh new file mode 100644 index 0000000000..c81853e5cb --- /dev/null +++ b/native-library-files/stp_project/clean_and_build_stp.sh @@ -0,0 +1,39 @@ +#/usr/bin/env/ bash + +# This removes raw STP buld files and rebuild it +# +set -e + +DESTN_DIR=$PWD/stpJ/install_include + +[ ! -f "$DESTN_DIR" ] && mkdir -p "$DESTN_DIR" || rm -rf "$DESTN_DIR"/* ||: + +[ -d "./stp/build" ] && rm -rf ./stp/build ||: +[ -d "./build" ] && rm -rf ./build ||: + +echo "Raw STP Build files removed " + +echo "Now copying necessary files to expend STP" +STP_HEADER=$PWD/stp/include/stp/ +yes | cp -f $PWD/stp_extend/c_interface.h $STP_HEADER +STP_INTERFACE_SRC=$PWD/stp/lib/Interface/ +yes | cp -f $PWD/stp_extend/ext_c_interface.cpp $STP_INTERFACE_SRC +yes | cp -f $PWD/stp_extend/CMakeLists.txt $STP_INTERFACE_SRC + +cd stp +mkdir ./build;cd ./build +echo +echo "Now rebuilding building STP ... " + + +cmake .. +make + +echo "Installing STP into $DESTN_DIR" +make DESTDIR=$DESTN_DIR install + +echo "... ... STP BUILD SUCCESSFULL" +ls +echo + +# cd ../../ diff --git a/native-library-files/stp_project/clean_clone_build_stp.sh b/native-library-files/stp_project/clean_clone_build_stp.sh new file mode 100644 index 0000000000..b42c8e7747 --- /dev/null +++ b/native-library-files/stp_project/clean_clone_build_stp.sh @@ -0,0 +1,42 @@ +#/usr/bin/env/ bash + +# This removes any existing STP repositories and build files +# and clones a new repo and build STP +# +set -e + +DESTN_DIR=$PWD/stpJ/install_include + +[ ! -f "$DESTN_DIR" ] && mkdir -p "$DESTN_DIR" || rm -rf "$DESTN_DIR"/* ||: + +[ -d "./stp" ] && rm -rf ./stp ||: +[ -d "./build" ] && rm -rf ./build ||: + +echo "Build files removed and STP repo deleted." +echo "Now cloning stp ..." + +git clone https://github.com/stp/stp.git + +echo "Now copying necessary files to expend STP" +STP_HEADER=$PWD/stp/include/stp/ +yes | cp -f $PWD/stp_extend/c_interface.h $STP_HEADER +STP_INTERFACE_SRC=$PWD/stp/lib/Interface/ +yes | cp -f $PWD/stp_extend/ext_c_interface.cpp $STP_INTERFACE_SRC +yes | cp -f $PWD/stp_extend/CMakeLists.txt $STP_INTERFACE_SRC + +cd stp +mkdir ./build;cd ./build +echo +echo "Now building STP ... " + +cmake .. +make + +echo "Installing STP into $DESTN_DIR" +make DESTDIR=$DESTN_DIR install + +echo "... ... STP BUILD SUCCESSFULL" +ls +echo + +# cd ../../ diff --git a/native-library-files/stp_project/clean_up.sh b/native-library-files/stp_project/clean_up.sh new file mode 100644 index 0000000000..9ca04c5bdc --- /dev/null +++ b/native-library-files/stp_project/clean_up.sh @@ -0,0 +1,23 @@ +#/usr/bin/env/ bash + +# Clean up the build files for StpJavaApi +# cryptominisat repo is also removed + +set -e + +# remove STP repo +#[ -d "./stp" ] && rm -rf ./stp ||: + + +# the built STP library should have been copied out +[ -d "./stpJ/build" ] && rm -rf ./stpJ/build ||: + +DESTN_DIR=$PWD/stpJ/install_include +[ ! -f "$DESTN_DIR" ] && rm -rf "$DESTN_DIR"/* ||: + +#remove cryptominisat repo +[ -d "./dependencies/cryptominisat" ] && rm -rf ./dependencies/cryptominisat ||: + +# java builds + +echo "Build files and cloned repos removed." diff --git a/native-library-files/stp_project/dependencies/install_depends.sh b/native-library-files/stp_project/dependencies/install_depends.sh new file mode 100644 index 0000000000..53e845e6c3 --- /dev/null +++ b/native-library-files/stp_project/dependencies/install_depends.sh @@ -0,0 +1,15 @@ +#!/usr/etc/ bash + +set -e +# prerequisite for building STP +# boost, flex, bison and minisat +sudo apt-get install cmake bison flex libboost-all-dev python perl minisat + +# CryptoMiniSat +git clone https://github.com/msoos/cryptominisat +cd cryptominisat +mkdir build && cd build +cmake .. +make +sudo make install +sudo ldconfig diff --git a/native-library-files/stp_project/install_prereq.sh b/native-library-files/stp_project/install_prereq.sh new file mode 100644 index 0000000000..074a6143ee --- /dev/null +++ b/native-library-files/stp_project/install_prereq.sh @@ -0,0 +1,7 @@ +#!/usr/etc/ bash + +# installs dependencies for building the STP library +set -e + +cd ./dependencies +sh install_depends.sh \ No newline at end of file diff --git a/native-library-files/stp_project/stpJ/CMakeLists.txt b/native-library-files/stp_project/stpJ/CMakeLists.txt new file mode 100644 index 0000000000..fd8e742efa --- /dev/null +++ b/native-library-files/stp_project/stpJ/CMakeLists.txt @@ -0,0 +1,95 @@ +cmake_minimum_required(VERSION 3.8.0) + +project(stpJ) + +#SWIG - load the cmake package and add to include set +find_package(SWIG REQUIRED) +include(UseSWIG) + +#JAVA & JNI - load the cmake package and add to include set +set(JAVA_AWT_INCLUDE_PATH NotNeeded) +find_package(Java REQUIRED) +find_package(JNI REQUIRED) +include(UseJava) + +include_directories(${JNI_INCLUDE_DIRS} ./include) # add jni to includes + +# variable setups +set(STP_SWIG_INTERFACE StpJavaApi.i) +set(CMAKE_SWIG_OUTPUTDIR "swig_builds") + +# find_library(OPENSMT_RAW_LIB "opensmt2" PATHS "./lib" NO_DEFAULT_PATH ) + +set(STP_DIR "" CACHE PATH "Try to use particular STP install (set this to folder where STPConfig.cmake is installed)") +find_package(STP REQUIRED) +message("Using STP library rooted at ${STP_DIR}") + +# Get full path to STP executable +get_target_property(STP_FULL_PATH ${STP_EXECUTABLE} LOCATION) +message("Full path to STP binary is ${STP_FULL_PATH}") + + +# JAVA package +set(CMAKE_SWIG_FLAGS -package org.sosy_lab.java_smt.solvers.stp) +set(CMAKE_SWIG_OUTDIR "org/sosy_lab/java_smt/solvers/stp") +# therefore +set(JAVA_API_DIR "build/org/sosy_lab/java_smt/solvers/stp") + +##################### END OF SET UP and VARIABLE SETTINGS ############### + +# /JavaSMT/native-library-files/opensmt2/opensmt/src/api +include_directories(${STP_INCLUDE_DIRS}) # ${CMAKE_CURRENT_SOURCE_DIR}/stp/build/include/) + +### USING SWIG: Build the C/C++ code into a library, accessed via our defined interface (in SWIG) + +# we are using C++ +# set_property(SOURCE ${STP_SWIG_INTERFACE} PROPERTY CPLUSPLUS ON) + +#using swig : we generate a interface to interact with the earlier created library +swig_add_library(stpJapi SHARED + LANGUAGE Java + SOURCES ${STP_SWIG_INTERFACE} ${CMAKE_CURRENT_SOURCE_DIR}/src/extStpSWIGapi.cpp + ) + +# here we link the library and the api we created for it together +swig_link_libraries(stpJapi PUBLIC stp) + +set_property(TARGET stpJapi PROPERTY SWIG_COMPILE_OPTIONS -Wall -g) + +target_include_directories(stpJapi PUBLIC + ${CMAKE_CURRENT_SOURCE_DIR}/include/ + ${CMAKE_CURRENT_SOURCE_DIR}/stp/build/include/stp/ +) +message("CURRENT SRC DIR IS ====> ${CMAKE_CURRENT_SOURCE_DIR}") +message("STP_INCLUDE_DIRS IS ====> ${STP_INCLUDE_DIRS}") + +### END of Build the Shared Library API + +# Build the library API now in JAVA - opensmt2JavaAPI.jar, linking the .so, and generated JNI java files +# For pyhton - swig_link_libraries(foobarPYapi foobarapi ${PYTHON_LIBRARIES} ) +# For JAVA : + +add_jar( + stpJavaAPI + SOURCES ${JAVA_API_DIR}/StpJavaApi.java + ${JAVA_API_DIR}/StpJavaApiJNI.java + ${JAVA_API_DIR}/exprkind_t.java + ${JAVA_API_DIR}/ifaceflag_t.java + ${JAVA_API_DIR}/type_t.java + ${JAVA_API_DIR}/Expr.java + ${JAVA_API_DIR}/Type.java + ${JAVA_API_DIR}/VC.java + ${JAVA_API_DIR}/WholeCounterExample.java +) + +FILE(GLOB_RECURSE ALL_JAVA_FILES ${JAVA_API_DIR}/*.java) + +# target_include_directories(stpJavaAPI PUBLIC +# ${CMAKE_CURRENT_SOURCE_DIR}/include/> +# ) + +#add_jar( +# stpJavaAPI +# SOURCES ${ALL_JAVA_FILES} +#) +add_dependencies( stpJavaAPI stpJapi ) diff --git a/native-library-files/stp_project/stpJ/StpJavaApi.i b/native-library-files/stp_project/stpJ/StpJavaApi.i new file mode 100644 index 0000000000..211a2552bf --- /dev/null +++ b/native-library-files/stp_project/stpJ/StpJavaApi.i @@ -0,0 +1,614 @@ +%module StpJavaApi + +//This is is where our API to STP is defined. We are basically customising +//the ../files/c_interface.h file in STP. The function names are unchanged. + +%{ // Prepocessor code + +// #ifndef _cvcl__include__c_interface_h_ +// #define _cvcl__include__c_interface_h_ +// #endif + +#include + +#include "extStpSWIGapi.h" + +// struct StpEnv; +// typedef struct StpEnv StpEnv; +// typedef void* VC; +// typedef void* Expr; +// typedef void* Type; +// typedef void* WholeCounterExample; + +int getNumOfAsserts(VC vc); + +void addAssertFormula(VC vc, Expr e); +void push(VC vc); +void pop(VC vc); +int checkSAT_old(VC vc); +int checkSAT(VC vc, Expr e); + +//adapted CPP Interface +StpEnv * createStpEnv(VC vc); +void destroyStpEnv(StpEnv * env); +int getCacheSize(StpEnv * env); +int getSymbolsSize(StpEnv * v); +void ext_push(StpEnv * env); +void ext_pop(StpEnv * env); +void ext_addFormula(StpEnv * env, Expr formula); +void ext_checkSat(StpEnv * env); + +const char * getAllModel(VC vc); +%} + +// Necessary extra includes +%include "typemaps.i" +%include "enums.swg" + +%typemap(javain) enum SWIGTYPE "$javainput.ordinal()" +%typemap(javaout) enum SWIGTYPE { + return $javaclassname.class.getEnumConstants()[$jnicall]; + } +%typemap(javabody) enum SWIGTYPE "" + + +///////////////////////////////////////////////////////////////////////////// +/// ENUMS +///////////////////////////////////////////////////////////////////////////// + +// %include "enums.swg" +// %javaconst(1); + +// Enum for Interface-only flags. +%inline %{ +enum ifaceflag_t +{ + EXPRDELETE, + MS, + SMS, + CMS4, + RISS, + MSP +}; + +//utility functions for this enum +void set_ifaceflag_t(enum ifaceflag_t h); +enum ifaceflag_t get_ifaceflag_t(); +%} + +// Enum that Covers all kinds of types that exist in STP. +%inline%{ +enum type_t +{ + BOOLEAN_TYPE, + BITVECTOR_TYPE, + ARRAY_TYPE, + UNKNOWN_TYPE +}; + +//utility functions for this enum +void set_type_t(enum type_t h); +enum type_t get_type_t(); +%} + +// Enum that Covers all kinds of expressions that exist in STP. +%inline%{ +enum exprkind_t +{ + UNDEFINED, + SYMBOL, + BVCONST, + BVNOT, + BVCONCAT, + BVOR, + BVAND, + BVXOR, + BVNAND, + BVNOR, + BVXNOR, + BVEXTRACT, + BVLEFTSHIFT, + BVRIGHTSHIFT, + BVSRSHIFT, + BVPLUS, + BVSUB, + BVUMINUS, + BVMULT, + BVDIV, + BVMOD, + SBVDIV, + SBVREM, + SBVMOD, + BVSX, + BVZX, + ITE, + BOOLEXTRACT, + BVLT, + BVLE, + BVGT, + BVGE, + BVSLT, + BVSLE, + BVSGT, + BVSGE, + EQ, + FALSE, + TRUE, + NOT, + AND, + OR, + NAND, + NOR, + XOR, + IFF, + IMPLIES, + PARAMBOOL, + READ, + WRITE, + ARRAY, + BITVECTOR, + BOOLEAN +}; + +//utility functions for this enum +void set_exprkind_t(enum exprkind_t h); +enum exprkind_t get_exprkind_t(); +%} + +///////////////////////////////////////////////////////////////////////////// +/// CLASS TYPES +///////////////////////////////////////////////////////////////////////////// + +// %{ +// typedef void* VC; +// typedef void* Expr; +// typedef void* Type; +// typedef void* WholeCounterExample; +// %} + +%nodefaultctor VC; +struct VC {}; + +%nodefaultctor Expr; +struct Expr {}; + +%nodefaultctor Type; +struct Type {}; + +%nodefaultctor WholeCounterExample; +struct WholeCounterExample {}; + +%nodefaultctor StpEnv; +struct StpEnv {}; + +/* +%inline%{ + +// TODO: hanldle this in Java Code; Remove if confirmed. Envoronment configs + struct Flags { + const char a; + const char c; + const char d; + const char m; + const char n; + const char p; + const char q; + const char r; + const char s; + const char t; + const char v; + const char w; + const char y; +} ProcessFlags ;= { + .a = 'a', + .c = 'c', + .d = 'd', + .m = 'm', + .n = 'n', + .p = 'p', + .q = 'q', + .r = 'r', + .s = 's', + .t = 't', + .v = 'v', + .w = 'w', + .y = 'y' +}; + +%} + */ + +%inline%{ + +///////////////////////////////////////////////////////////////////////////// +/// EXTENTION ATTEMPTS +///////////////////////////////////////////////////////////////////////////// + +int extraFunctionSum(int a, int b); + +void extraSumUpto(int num); + +void ext_AssertFormula(VC vc, Expr e); + +int getNumOfAsserts(VC vc); + +int getSomePrinting(VC vc); + +char* getSomeXter(VC vc); + +void addAssertFormula(VC vc, Expr e); +void push(VC vc); +void pop(VC vc); +int checkSAT_old(VC vc); + +int checkSAT(VC vc, Expr e); + +const char * getAllModel(VC vc); + +//CPP +StpEnv * createStpEnv(VC vc); +void destroyStpEnv(StpEnv * env); +int getCacheSize(StpEnv * env); +int getSymbolsSize(StpEnv * v); +void ext_push(StpEnv * env); +void ext_pop(StpEnv * env); +void ext_addFormula(StpEnv * env, Expr formula); +void ext_checkSat(StpEnv * env); + +///////////////////////////////////////////////////////////////////////////// +/// API INITIALISATION AND CONFIG +///////////////////////////////////////////////////////////////////////////// + +const char* get_git_version_sha(void); + +const char* get_git_version_tag(void); + +const char* get_compilation_env(void); + +void process_argument(const char ch, VC bm); + +void vc_setInterfaceFlags(VC vc, enum ifaceflag_t f, int param_value); + +VC vc_createValidityChecker(void); + +Type vc_boolType(VC vc); + +Type vc_arrayType(VC vc, Type typeIndex, Type typeData); + +///////////////////////////////////////////////////////////////////////////// +/// EXPR MANUPULATION METHODS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_varExpr(VC vc, const char* name, Type type); + +//Expr vc_varExpr1(VC vc, const char* name, int indexwidth, +// int valuewidth); + +Type vc_getType(VC vc, Expr e); + +int vc_getBVLength(VC vc, Expr e); + +Expr vc_eqExpr(VC vc, Expr child0, Expr child1); + +///////////////////////////////////////////////////////////////////////////// +/// BOOLEAN EXPRESSIONS +/// +/// The following functions create boolean expressions. +/// The children provided as arguments must be of type boolean. +/// +/// An exception is the function vc_iteExpr(). +/// In the case of vc_iteExpr() the conditional must always be boolean, +/// but the thenExpr (resp. elseExpr) can be bit-vector or boolean type. +/// However, the thenExpr and elseExpr must be both of the same type. +/// +///////////////////////////////////////////////////////////////////////////// + +Expr vc_trueExpr(VC vc); + +Expr vc_falseExpr(VC vc); + +Expr vc_notExpr(VC vc, Expr child); + +Expr vc_andExpr(VC vc, Expr left, Expr right); + +Expr vc_andExprN(VC vc, Expr* children, int numOfChildNodes); + +Expr vc_orExpr(VC vc, Expr left, Expr right); + +Expr vc_orExprN(VC vc, Expr* children, int numOfChildNodes); + +Expr vc_xorExpr(VC vc, Expr left, Expr right); + +Expr vc_impliesExpr(VC vc, Expr hyp, Expr conc); + +Expr vc_iffExpr(VC vc, Expr left, Expr right); + +Expr vc_iteExpr(VC vc, Expr conditional, Expr thenExpr, Expr elseExpr); + +Expr vc_boolToBVExpr(VC vc, Expr form); + +Expr vc_paramBoolExpr(VC vc, Expr var, Expr param); + +///////////////////////////////////////////////////////////////////////////// +/// ARRAY EXPRESSIONS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_readExpr(VC vc, Expr array, Expr index); + +Expr vc_writeExpr(VC vc, Expr array, Expr index, Expr newValue); + +Expr vc_parseExpr(VC vc, const char* filepath); + +void vc_printExpr(VC vc, Expr e); + +void vc_printExprCCode(VC vc, Expr e); + +char* vc_printSMTLIB(VC vc, Expr e); + +void vc_printExprFile(VC vc, Expr e, int fd); + +void vc_printStateToBuffer(VC vc, char **buf, unsigned long *len); + +void vc_printExprToBuffer(VC vc, Expr e, char** buf, + unsigned long* len); + +void vc_printCounterExample(VC vc); + +void vc_printVarDecls(VC vc); + +void vc_clearDecls(VC vc); + +// void vc_printAsserts(VC vc, int simplify_print _CVCL_DEFAULT_ARG(0)); +void vc_printAsserts(VC vc, int simplify_print); + + +void vc_printQueryStateToBuffer(VC vc, Expr e, char** buf, + unsigned long* len, + int simplify_print); + +void vc_printCounterExampleToBuffer(VC vc, char** buf, unsigned long* len); + +void vc_printQuery(VC vc); + +///////////////////////////////////////////////////////////////////////////// +/// CONTEXT RELATED METHODS +///////////////////////////////////////////////////////////////////////////// + + +void vc_assertFormula(VC vc, Expr e); + +Expr vc_simplify(VC vc, Expr e); + +int vc_query_with_timeout(VC vc, Expr e, int timeout_max_conflicts, int timeout_max_time); + +int vc_query(VC vc, Expr e); + +Expr vc_getCounterExample(VC vc, Expr e); + +void vc_getCounterExampleArray(VC vc, Expr e, Expr** outIndices, + Expr** outValues, int* outSize); + +int vc_counterexample_size(VC vc); + +void vc_push(VC vc); + +void vc_pop(VC vc); + +int getBVInt(Expr e); + +unsigned int getBVUnsigned(Expr e); + +unsigned long long int getBVUnsignedLongLong(Expr e); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +Type vc_bvType(VC vc, int no_bits); + +Type vc_bv32Type(VC vc); + +//Const expressions for string, int, long-long, etc + +Expr vc_bvConstExprFromDecStr(VC vc, int width, + const char* decimalInput); + +Expr vc_bvConstExprFromStr(VC vc, const char* binaryInput); + +Expr vc_bvConstExprFromInt(VC vc, int bitWidth, unsigned int value); + +Expr vc_bvConstExprFromLL(VC vc, int bitWidth, + unsigned long long value); + +Expr vc_bv32ConstExprFromInt(VC vc, unsigned int value); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR ARITHMETIC OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_bvConcatExpr(VC vc, Expr left, Expr right); + +Expr vc_bvPlusExpr(VC vc, int bitWidth, Expr left, Expr right); + +Expr vc_bvPlusExprN(VC vc, int bitWidth, Expr* children, + int numOfChildNodes); + +Expr vc_bv32PlusExpr(VC vc, Expr left, Expr right); + +Expr vc_bvMinusExpr(VC vc, int bitWidth, Expr left, Expr right); + +Expr vc_bv32MinusExpr(VC vc, Expr left, Expr right); + +Expr vc_bvMultExpr(VC vc, int bitWidth, Expr left, Expr right); + +Expr vc_bv32MultExpr(VC vc, Expr left, Expr right); + +Expr vc_bvDivExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +Expr vc_bvModExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +Expr vc_sbvDivExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +Expr vc_sbvModExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +Expr vc_sbvRemExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR COMPARISON OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_bvLtExpr(VC vc, Expr left, Expr right); + +Expr vc_bvLeExpr(VC vc, Expr left, Expr right); + +Expr vc_bvGtExpr(VC vc, Expr left, Expr right); + +Expr vc_bvGeExpr(VC vc, Expr left, Expr right); + +Expr vc_sbvLtExpr(VC vc, Expr left, Expr right); + +Expr vc_sbvLeExpr(VC vc, Expr left, Expr right); + +Expr vc_sbvGtExpr(VC vc, Expr left, Expr right); + +Expr vc_sbvGeExpr(VC vc, Expr left, Expr right); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR BITWISE OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_bvUMinusExpr(VC vc, Expr child); + +Expr vc_bvAndExpr(VC vc, Expr left, Expr right); + +Expr vc_bvOrExpr(VC vc, Expr left, Expr right); + +Expr vc_bvXorExpr(VC vc, Expr left, Expr right); + +Expr vc_bvNotExpr(VC vc, Expr child); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR SHIFT OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +Expr vc_bvLeftShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +Expr vc_bvRightShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +Expr vc_bvSignedRightShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR EXTRACTION & EXTENSION +///////////////////////////////////////////////////////////////////////////// + +Expr vc_bvExtract(VC vc, Expr child, int high_bit_no, + int low_bit_no); + +Expr vc_bvBoolExtract_Zero(VC vc, Expr x, int bit_no); + +Expr vc_bvBoolExtract_One(VC vc, Expr x, int bit_no); + +Expr vc_bvSignExtend(VC vc, Expr child, int newWidth); + +///////////////////////////////////////////////////////////////////////////// +/// CONVENIENCE FUNCTIONS FOR ARRAYS +///////////////////////////////////////////////////////////////////////////// + +/*C pointer support: C interface to support C memory arrays in CVCL */ + +Expr vc_bvCreateMemoryArray(VC vc, const char* arrayName); + +Expr vc_bvReadMemoryArray(VC vc, Expr array, Expr byteIndex, + int numOfBytes); + +Expr vc_bvWriteToMemoryArray(VC vc, Expr array, Expr byteIndex, + Expr element, int numOfBytes); + +///////////////////////////////////////////////////////////////////////////// +/// GENERAL EXPRESSION OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +char* exprString(Expr e); + +char* typeString(Type t); + +Expr getChild(Expr e, int n); + +// This function is wrong +//int vc_isBool(Expr e); + +void vc_registerErrorHandler(void (*error_hdlr)(const char* err_msg)); + + +// ERROR HANDLER ToDo: Good but Error Handling here needs rework + +//%callback("%(uppercase)s"); +//void errorHandler(const char * err_msg) +//{ +// printf("Error: %s\n", err_msg); +// exit(1); +//} +//%nocallback; + + +int vc_getHashQueryStateToBuffer(VC vc, Expr query); + +void vc_Destroy(VC vc); + +void vc_DeleteExpr(Expr e); + +WholeCounterExample vc_getWholeCounterExample(VC vc); + +Expr vc_getTermFromCounterExample(VC vc, Expr e, WholeCounterExample c); + +void vc_deleteWholeCounterExample(WholeCounterExample cc); + +enum exprkind_t getExprKind(Expr e); + +int getDegree(Expr e); + +int getBVLength(Expr e); + +enum type_t getType(Expr e); + +// get value bit width + +int getVWidth(Expr e); + +int getIWidth(Expr e); + +void vc_printCounterExampleFile(VC vc, int fd); + +const char* exprName(Expr e); + +int getExprID(Expr ex); + +int vc_parseMemExpr(VC vc, const char* s, Expr* outQuery, + Expr* outAsserts); + +bool vc_supportsMinisat(VC vc); + +bool vc_useMinisat(VC vc); + +bool vc_isUsingMinisat(VC vc); + +bool vc_supportsSimplifyingMinisat(VC vc); + +bool vc_useSimplifyingMinisat(VC vc); + +bool vc_supportsCryptominisat(VC vc); + +bool vc_useCryptominisat(VC vc); + +bool vc_isUsingCryptominisat(VC vc); + +bool vc_supportsRiss(VC vc); + +bool vc_useRiss(VC vc); + +bool vc_isUsingRiss(VC vc); + +%} \ No newline at end of file diff --git a/native-library-files/stp_project/stpJ/include/extStpSWIGapi.h b/native-library-files/stp_project/stpJ/include/extStpSWIGapi.h new file mode 100644 index 0000000000..60ceecdf0e --- /dev/null +++ b/native-library-files/stp_project/stpJ/include/extStpSWIGapi.h @@ -0,0 +1,67 @@ + + +// #ifndef _cvcl__include__c_interface_h_ +// #define _cvcl__include__c_interface_h_ + + +#ifdef __cplusplus +#define _CVCL_DEFAULT_ARG(v) = v +#else +#define _CVCL_DEFAULT_ARG(v) +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +#include + +// #include + +// #include "stp/c_interface.h" + +// #include +// #include +// #include + +// #include "stp/Interface/fdstream.h" +// #include "stp/Parser/parser.h" +// #include "stp/Printer/printers.h" +// #include "stp/cpp_interface.h" +// #include "stp/Util/GitSHA1.h" +// FIXME: External library +// #include "extlib-abc/cnf_short.h" + +// using std::cout; +// using std::ostream; +// using std::stringstream; +// using std::string; +// using std::fdostream; +// using std::endl; + +// typedef void* VC; +// typedef void* Expr; +// typedef void* Type; +// typedef void* WholeCounterExample; + +//CPP +struct StpEnv; +typedef struct StpEnv StpEnv; +typedef void* VC; +typedef void* Expr; +typedef void* Type; +typedef void* WholeCounterExample; + +int getNumOfAsserts(VC vc); + +#ifdef __cplusplus +} +#endif + +// #undef DLL_PUBLIC // Undefine internal macro to prevent it from leaking into the API. +// #undef DLL_LOCAL // Undefine internal macro to prevent it from leaking into the API. + +#undef _CVCL_DEFAULT_ARG // Undefine macro to not pollute global macro namespace! + +// #endif // _cvcl__include__c_interface_h_ + diff --git a/native-library-files/stp_project/stpJ/src/extStpSWIGapi.cpp b/native-library-files/stp_project/stpJ/src/extStpSWIGapi.cpp new file mode 100644 index 0000000000..342d771def --- /dev/null +++ b/native-library-files/stp_project/stpJ/src/extStpSWIGapi.cpp @@ -0,0 +1,14 @@ +#include "../include/extStpSWIGapi.h" + +/* + Normally c_interface.h should be extended here calling the cpp_interface.h + However cpp_interface.h is not included in build files (STRANGE !!!) + A Solution: + Create ext_c_interface.cpp right inside STP and extend it there + then the extended functions can be part of the SWIG interface + */ + +int getNumOfAsserts(VC vc) +{ + return 0; +} diff --git a/native-library-files/stp_project/stp_extend/CMakeLists.txt b/native-library-files/stp_project/stp_extend/CMakeLists.txt new file mode 100644 index 0000000000..51f708b68b --- /dev/null +++ b/native-library-files/stp_project/stp_extend/CMakeLists.txt @@ -0,0 +1,47 @@ +# AUTHORS: Dan Liew, Ryan Gvostes, Mate Soos +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. + + +# FIXME: Merge into one library +add_library(cinterface OBJECT + c_interface.cpp + ext_c_interface.cpp +) + +add_library(cppinterface OBJECT + cpp_interface.cpp +) + +add_dependencies(cinterface ASTKind_header) +add_dependencies(cppinterface ASTKind_header) + +# FIXME: Remove this +# For legacy reasons make the interface header files available in include/stp/ in the build directory +if (NOT ("${PROJECT_BINARY_DIR}" STREQUAL "${PROJECT_SOURCE_DIR}")) + add_custom_target(CopyPublicHeaders ALL) + set(HEADER_DEST "${PROJECT_BINARY_DIR}/include/stp") + foreach(public_header c_interface.h cpp_interface.h) + add_custom_command(TARGET CopyPublicHeaders PRE_BUILD + COMMAND ${CMAKE_COMMAND} -E make_directory ${HEADER_DEST} + COMMAND ${CMAKE_COMMAND} -E echo "LEGACY: Copying ${public_header} to ${HEADER_DEST}" + COMMAND ${CMAKE_COMMAND} -E copy_if_different "${PROJECT_SOURCE_DIR}/include/stp/${public_header}" "${HEADER_DEST}/${public_header}" + ) + endforeach() +endif() diff --git a/native-library-files/stp_project/stp_extend/c_interface.h b/native-library-files/stp_project/stp_extend/c_interface.h new file mode 100644 index 0000000000..40d2a58678 --- /dev/null +++ b/native-library-files/stp_project/stp_extend/c_interface.h @@ -0,0 +1,1238 @@ +/******************************************************************** + * AUTHORS: Michael Katelman, Vijay Ganesh, Trevor Hansen, Andrew V. Jones + * + * BEGIN DATE: Apr, 2008 + * +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in +all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +THE SOFTWARE. +********************************************************************/ + +#ifndef _cvcl__include__c_interface_h_ +#define _cvcl__include__c_interface_h_ + +#ifdef __cplusplus +#define _CVCL_DEFAULT_ARG(v) = v +#else +#define _CVCL_DEFAULT_ARG(v) +#endif + +#ifdef __cplusplus +extern "C" { +#endif + +#include + +///////////////////////////////////////////////////////////////////////////// +/// STP API INTERNAL MACROS FOR LINKING +/// +/// These are undefined at the end of this file to prevent them from leaking +/// into code that includes it. +///////////////////////////////////////////////////////////////////////////// + +#if defined(_MSC_VER) +// NOTE: for now, we need STP_SHARED_LIB for clients of the statically linked +// STP library, for which linking fails when DLL_PUBLIC is __declspec(dllimport). +#if defined(STP_SHARED_LIB) && defined(STP_EXPORTS) +// This is visible when building the STP library as a DLL. +#define DLL_PUBLIC __declspec(dllexport) +#elif defined(STP_SHARED_LIB) +// This is visible for STP clients. +#define DLL_PUBLIC __declspec(dllimport) +#else +#define DLL_PUBLIC +#endif + +// Symbols are hidden by default in MSVC. +#define DLL_LOCAL + +#elif defined(__GNUC__) || defined(__clang__) +#define DLL_PUBLIC __attribute__((visibility("default"))) +#define DLL_LOCAL __attribute__((visibility("hidden"))) +#else +#define DLL_PUBLIC +#define DLL_LOCAL +#endif + +///////////////////////////////////////////////////////////////////////////// +/// STP API Types +/// +/// This gives absolutely no pointer typing at compile-time. Most C +/// users prefer this over stronger typing. User is the king. A +/// stronger typed interface is in the works. +///////////////////////////////////////////////////////////////////////////// + +#ifdef STP_STRONG_TYPING // not used for now! +#else +typedef void* VC; +typedef void* Expr; +typedef void* Type; +typedef void* WholeCounterExample; +#endif + +//////////////////// EXTENSION //////////// + +void addAssertFormula(VC vc, Expr e); +void push(VC vc); +void pop(VC vc); +int checkSAT_old(VC vc); +int checkSAT(VC vc, Expr e); + +//adapted CPP Interface +struct StpEnv; +typedef struct StpEnv StpEnv; +StpEnv * createStpEnv(VC vc); +void destroyStpEnv(StpEnv * env); +int getCacheSize(StpEnv * env); +int getSymbolsSize(StpEnv * v); +void ext_push(StpEnv * env); +void ext_pop(StpEnv * env); +void ext_addFormula(StpEnv * env, Expr formula); +void ext_checkSat(StpEnv * env); + +const char * getAllModel(VC vc); + +//////////////////// END OF EXTENSION //////////// + +///////////////////////////////////////////////////////////////////////////// +/// START API +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns the C string for the git sha of STP +//! +DLL_PUBLIC const char* get_git_version_sha(void); + +//! \brief Returns the C string for the git tag of STP +//! +DLL_PUBLIC const char* get_git_version_tag(void); + +//! \brief Returns the C string for the compilation env of STP +//! +DLL_PUBLIC const char* get_compilation_env(void); + +//! \brief Processes the given flag represented as char for the given validity checker. +//! +//! The following flags are supported: +//! - 'a': Disables optimization. TODO: What kind of optimization is meant here? +//! - 'c': Enables construction of counter examples. +//! - 'd': Enables construction and checking of counter examples. Superseeds flag 'c'. +//! - 'm': Use SMTLib1 parser. Conflicts with using SMTLib2 parser. +//! - 'n': Enables printing of the output. TODO: What is meant with output here? +//! - 'p': Enables printing of counter examples. +//! - 'q': Enables printing of array values in declared order. +//! - 'r': Enables accermannisation. +//! - 's': Sets the status flag to true. TODO: What consequenses does this have? +//! - 't': Enables quick statistics. TODO: What is this? +//! - 'v': Enables printing of nodes. +//! - 'w': Enables word-level solving. TODO: What is mean with this? +//! - 'y': Enables printing binaries. TODO: What is meant with this? +//! +//! This function panics if given an unsupported or unknown flag. +//! +DLL_PUBLIC void process_argument(const char ch, VC bm); + +//! \brief Deprecated: use process_argument instead! +//! +//! Sets flags for the validity checker. +//! For more information about this look into the documentation of process_argument. +//! +//! Parameter num_absrefine has no effect in the current implementation. +//! It is left for compatibility with existing code. +//! +DLL_PUBLIC void vc_setFlags(VC vc, char c, + int num_absrefine _CVCL_DEFAULT_ARG(0)); + +//! \brief Deprecated: use process_argument instead! +//! +//! Sets flags for the validity checker. +//! For more information about this look into the documentation of process_argument. +//! +DLL_PUBLIC void vc_setFlag(VC vc, char c); + +//! Interface-only flags. +//! +enum ifaceflag_t +{ + //! Tells the validity checker that it is responsible for resource + //! deallocation of its allocated expressions. + //! + //! This is set to true by default. + //! + //! Affected methods are: + //! - vc_arrayType + //! - vc_boolType + //! - vc_bvType + //! - vc_bv32Type + //! - vc_vcConstExprFromInt + //! + //! Changing this flag while STP is running may result in undefined behaviour. + //! + //! Use this with great care; otherwise memory leaks are very easily possible! + //! + EXPRDELETE, + + //! Use the minisat SAT solver. + //! + MS, + + //! Use a simplifying version of the minisat SAT solver. + //! + SMS, + + //! Use the crypto minisat version 4 or higher (currently version 5) solver. + //! + CMS4, + + //! Use the SAT solver Riss. + //! + RISS, + + //! \brief Deprecated: use `MS` instead! + //! + //! This used to be the array version of the minisat SAT solver. + //! + //! Currently simply forwards to MS. + //! + MSP + +}; + +//! \brief Sets the given interface flag for the given validity checker to param_value. +//! +//! Use this to set the underlying SAT solver used by STP or to change +//! the global behaviour for expression ownership semantics via EXPRDELETE. +//! +DLL_PUBLIC void vc_setInterfaceFlags(VC vc, enum ifaceflag_t f, + int param_value); + +//! \brief Deprecated: this functionality is no longer needed! +//! +//! Since recent versions of STP division is always total. +DLL_PUBLIC void make_division_total(VC vc); + +//! \brief Creates a new instance of an STP validity checker. +//! +//! Validity checker is the context for all STP resources like expressions, +//! type and counter examples that may be generated while running STP. +//! +//! It is also the interface for assertions and queries. +//! +DLL_PUBLIC VC vc_createValidityChecker(void); + +//! \brief Returns the boolean type for the given validity checker. +//! +DLL_PUBLIC Type vc_boolType(VC vc); + +//! \brief Returns an array type with the given index type and data type +//! for the given validity checker. +//! +//! Note that index type and data type must both be of bitvector (bv) type. +//! +DLL_PUBLIC Type vc_arrayType(VC vc, Type typeIndex, Type typeData); + +///////////////////////////////////////////////////////////////////////////// +/// EXPR MANUPULATION METHODS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a variable (symbol) expression with the given name and type. +//! +//! The type cannot be a function type. (TODO: Are function type still a thing in STP?) +//! +//! The variable name must only consist of alphanumerics and underscore +//! characters, otherwise this may behave in undefined ways, e.g. segfault. +//! +DLL_PUBLIC Expr vc_varExpr(VC vc, const char* name, Type type); + +//! \brief Similar to vc_varExpr but more bare metal. Do not use this unless +//! you really know what you are doing! +//! +//! Note: This should be deprecated in favor of the saner vc_varExpr API +//! and as this API leaks implementation details of STP. +//! +//! The variable name must only consist of alphanumerics and underscore +//! characters, otherwise this may behave in undefined ways, e.g. segfault. +//! +DLL_PUBLIC Expr vc_varExpr1(VC vc, const char* name, int indexwidth, + int valuewidth); + +//! \brief Returns the type of the given expression. +//! +DLL_PUBLIC Type vc_getType(VC vc, Expr e); + +//! \brief Returns the bit-width of the given bitvector. +//! +DLL_PUBLIC int vc_getBVLength(VC vc, Expr e); + +//! \brief Create an equality expression. The two children must have the same type. +//! +//! Returns a boolean expression. +//! +DLL_PUBLIC Expr vc_eqExpr(VC vc, Expr child0, Expr child1); + +///////////////////////////////////////////////////////////////////////////// +/// BOOLEAN EXPRESSIONS +/// +/// The following functions create boolean expressions. +/// The children provided as arguments must be of type boolean. +/// +/// An exception is the function vc_iteExpr(). +/// In the case of vc_iteExpr() the conditional must always be boolean, +/// but the thenExpr (resp. elseExpr) can be bit-vector or boolean type. +/// However, the thenExpr and elseExpr must be both of the same type. +/// +///////////////////////////////////////////////////////////////////////////// + +//! \brief Creates a boolean expression that represents true. +//! +DLL_PUBLIC Expr vc_trueExpr(VC vc); + +//! \brief Creates a boolean expression that represents false. +//! +DLL_PUBLIC Expr vc_falseExpr(VC vc); + +//! \brief Creates a boolean not expression that logically negates its child. +//! +DLL_PUBLIC Expr vc_notExpr(VC vc, Expr child); + +//! \brief Creates a binary and-expression that represents a conjunction +//! of the given boolean child expressions. +//! +DLL_PUBLIC Expr vc_andExpr(VC vc, Expr left, Expr right); + +//! \brief Creates an and-expression with multiple child boolean expressions +//! that represents the conjunction of all of its child expressions. +//! +//! This API is useful since SMTLib2 defines non-binary expressions for logical-and. +//! +DLL_PUBLIC Expr vc_andExprN(VC vc, Expr* children, int numOfChildNodes); + +//! \brief Creates a binary or-expression that represents a disjunction +//! of the given boolean child expressions. +//! +DLL_PUBLIC Expr vc_orExpr(VC vc, Expr left, Expr right); + +//! \brief Creates an or-expression with multiple child boolean expressions +//! that represents the disjunction of all of its child expressions. +//! +//! This API is useful since SMTLib2 defines non-binary expressions for logical-or. +//! +DLL_PUBLIC Expr vc_orExprN(VC vc, Expr* children, int numOfChildNodes); + +//! \brief Creates a binary xor-expressions for the given boolean child expressions. +//! +DLL_PUBLIC Expr vc_xorExpr(VC vc, Expr left, Expr right); + +//! \brief Creates an implies-expression for the given hyp (hypothesis) and +//! conc (conclusion) boolean expressions. +//! +DLL_PUBLIC Expr vc_impliesExpr(VC vc, Expr hyp, Expr conc); + +//! \brief Creates an if-and-only-if-expression for the given boolean expressions. +//! +DLL_PUBLIC Expr vc_iffExpr(VC vc, Expr left, Expr right); + +//! \brief Creates an if-then-else-expression for the given conditional boolean expression +//! and its then and else expressions which must be of the same type. +//! +//! The output type of this API may be of boolean or bitvector type. +//! +DLL_PUBLIC Expr vc_iteExpr(VC vc, Expr conditional, Expr thenExpr, + Expr elseExpr); + +//! \brief Returns a bitvector expression from the given boolean expression. +//! +//! Returns a constant bitvector expression that represents one (1) if +//! the given boolean expression was false or returns a bitvector expression +//! representing zero (0) otherwise. +//! +//! Panics if the given expression is not of boolean type. +//! +DLL_PUBLIC Expr vc_boolToBVExpr(VC vc, Expr form); + +//! \brief Creates a parameterized boolean expression with the given boolean +//! variable expression and the parameter param. +//! +DLL_PUBLIC Expr vc_paramBoolExpr(VC vc, Expr var, Expr param); + +///////////////////////////////////////////////////////////////////////////// +/// ARRAY EXPRESSIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns an array-read-expression representing the reading of +//! the given array's entry of the given index. +//! +//! The array parameter must be of type array and index must be of type bitvector. +//! +DLL_PUBLIC Expr vc_readExpr(VC vc, Expr array, Expr index); + +//! \brief Returns an array-write-expressions representing the writing of +//! the given new value into the given array at the given entry index. +//! +//! The array parameter must be of type array, and index and newValue of type bitvector. +//! +DLL_PUBLIC Expr vc_writeExpr(VC vc, Expr array, Expr index, Expr newValue); + +//! \brief Parses the expression stored in the file of the given filepath +//! and returns it on success. +//! +//! TODO: What format is expected? SMTLib2? +//! Does the user have to deallocate resources for the returned expression? +//! Why exactly is this "pretty cool!"? +//! +DLL_PUBLIC Expr vc_parseExpr(VC vc, const char* filepath); + +//! \brief Prints the given expression to stdout in the presentation language. +//! +DLL_PUBLIC void vc_printExpr(VC vc, Expr e); + +//! \brief Prints the given expression to stdout as C code. +//! +DLL_PUBLIC void vc_printExprCCode(VC vc, Expr e); + +//! \brief Prints the given expression to stdout in the STMLib2 format. +//! +DLL_PUBLIC char* vc_printSMTLIB(VC vc, Expr e); + +//! \brief Prints the given expression into the file with the given file descriptor +//! in the presentation language. +//! +DLL_PUBLIC void vc_printExprFile(VC vc, Expr e, int fd); + +// //! \brief Prints the state of the given validity checker into +// //! buffer allocated by STP stores it into the given 'buf' alongside +// //! its length into 'len'. +// //! +// //! It is the responsibility of the caller to free the buffer. +// //! +// void vc_printStateToBuffer(VC vc, char **buf, unsigned long *len); + +//! \brief Prints the given expression into a buffer allocated by STP. +//! +//! The buffer is returned via output parameter 'buf' alongside its length 'len'. +//! It is the responsibility of the caller to free the memory afterwards. +DLL_PUBLIC void vc_printExprToBuffer(VC vc, Expr e, char** buf, + unsigned long* len); + +//! \brief Prints the counter example after an invalid query to stdout. +//! +//! This method should only be called after a query which returns false. +//! +DLL_PUBLIC void vc_printCounterExample(VC vc); + +//! \brief Prints variable declarations to stdout. +//! +DLL_PUBLIC void vc_printVarDecls(VC vc); + +//! \brief Clears the internal list of variables that are maintained +//! for printing purposes via 'vc_printVarDecls'. +//! +//! A user may want to do this after finishing printing the variable +//! declarations to prevent memory leaks. +//! This is also useful if printing of declarations is never wanted. +//! +DLL_PUBLIC void vc_clearDecls(VC vc); + +//! \brief Prints assertions to stdout. +//! +//! The validity checker's flag 'simplify_print' must be set to '1' +//! to enable simplifications of the asserted formulas during printing. +//! +DLL_PUBLIC void vc_printAsserts(VC vc, int simplify_print _CVCL_DEFAULT_ARG(0)); + +//! \brief Prints the state of the query to a buffer allocated by STP +//! that is returned via output parameter 'buf' alongside its +//! length in 'len'. +//! +//! It is the callers responsibility to free the buffer's memory. +//! +//! The validity checker's flag 'simplify_print' must be set to '1' +//! to enable simplifications of the query state during printing. +//! +DLL_PUBLIC void vc_printQueryStateToBuffer(VC vc, Expr e, char** buf, + unsigned long* len, + int simplify_print); + +//! \brief Prints the found counter example to a buffer allocated by STP +//! that is returned via output parameter 'buf' alongside its +//! length in 'len'. +//! +//! It is the callers responsibility to free the buffer's memory. +//! +//! The validity checker's flag 'simplify_print' must be set to '1' +//! to enable simplifications of the counter example during printing. +//! +DLL_PUBLIC void vc_printCounterExampleToBuffer(VC vc, char** buf, + unsigned long* len); + +//! \brief Prints the query to stdout in presentation language. +//! +DLL_PUBLIC void vc_printQuery(VC vc); + +///////////////////////////////////////////////////////////////////////////// +/// CONTEXT RELATED METHODS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Adds the given expression as assertion to the given validity checker. +//! +//! The expression must be of type boolean. +//! +DLL_PUBLIC void vc_assertFormula(VC vc, Expr e); + +//! \brief Simplifies the given expression with respect to the given validity checker. +//! +DLL_PUBLIC Expr vc_simplify(VC vc, Expr e); + +//! \brief Checks the validity of the given expression 'e' in the given context. +//! +//! 'timeout_max_conflicts' is represented and expected as the number of conflicts +//! 'timeout_max_time' is represented and expected in seconds. +//! The given expression 'e' must be of type boolean. +//! +//! Returns ... +//! 0: if 'e' is INVALID +//! 1: if 'e' is VALID +//! 2: if errors occured +//! 3: if the timeout was reached +//! +//! Note: Only the cryptominisat solver supports timeout_max_time +//! +DLL_PUBLIC int vc_query_with_timeout(VC vc, Expr e, int timeout_max_conflicts, int timeout_max_time); + +//! \brief Checks the validity of the given expression 'e' in the given context +//! with an unlimited timeout. +//! +//! This simply forwards to 'vc_query_with_timeout'. +//! +//! Note: Read the documentation of 'vc_query_with_timeout' for more information +//! about subtle details. +//! +DLL_PUBLIC int vc_query(VC vc, Expr e); + +//! \brief Returns the counter example after an invalid query. +//! +DLL_PUBLIC Expr vc_getCounterExample(VC vc, Expr e); + +//! \brief Returns an array from a counter example after an invalid query. +//! +//! The buffer for the array is allocated by STP and returned via the +//! non-null expected out parameters 'outIndices' for the indices, 'outValues' +//! for the values and 'outSize' for the size of the array. +//! +//! It is the caller's responsibility to free the memory afterwards. +//! +DLL_PUBLIC void vc_getCounterExampleArray(VC vc, Expr e, Expr** outIndices, + Expr** outValues, int* outSize); + +//! \brief Returns the size of the counter example array, +//! i.e. the number of variable and array locations +//! in the counter example. +//! +DLL_PUBLIC int vc_counterexample_size(VC vc); + +//! \brief Checkpoints the current context and increases the scope level. +//! +//! TODO: What effects has this? +//! +DLL_PUBLIC void vc_push(VC vc); + +//! \brief Restores the current context to its state at the last checkpoint. +//! +//! TODO: What effects has this? +//! +DLL_PUBLIC void vc_pop(VC vc); + +//! \brief Returns the associated integer from the given bitvector expression. +//! +//! Panics if the given bitvector cannot be represented by an 'int'. +//! +DLL_PUBLIC int getBVInt(Expr e); + +//! \brief Returns the associated unsigned integer from the given bitvector expression. +//! +//! Panics if the given bitvector cannot be represented by an 'unsigned int'. +//! +DLL_PUBLIC unsigned int getBVUnsigned(Expr e); + +//! Return an unsigned long long int from a constant bitvector expressions + +//! \brief Returns the associated unsigned long long integer from the given bitvector expression. +//! +//! Panics if the given bitvector cannot be represented by an 'unsigned long long int'. +//! +DLL_PUBLIC unsigned long long int getBVUnsignedLongLong(Expr e); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns the bitvector type for the given validity checker. +//! +DLL_PUBLIC Type vc_bvType(VC vc, int no_bits); + +//! \brief Returns the bitvector type with a bit-width of 32 for the +//! given validity checker. +//! +//! This is equal to calling 'vc_bvType(vc, 32)'. +//! +//! Note: This is a convenience function that simply forwards its input. +//! +DLL_PUBLIC Type vc_bv32Type(VC vc); + +//Const expressions for string, int, long-long, etc + +//! \brief Parses the given string and returns an associated bitvector expression. +//! +//! This function expects the input string to be of decimal format. +//! +DLL_PUBLIC Expr vc_bvConstExprFromDecStr(VC vc, int width, + const char* decimalInput); + +//! \brief Parses the given string and returns an associated bitvector expression. +//! +//! This function expects the input string to be of binary format. +//! +DLL_PUBLIC Expr vc_bvConstExprFromStr(VC vc, const char* binaryInput); + +//! \brief Returns a bitvector with 'bitWidth' bit-width from the given +//! unsigned integer value. +//! +//! The 'bitWidth' must be large enough to fully store the given value's bit representation. +//! +DLL_PUBLIC Expr vc_bvConstExprFromInt(VC vc, int bitWidth, unsigned int value); + +//! \brief Returns a bitvector with 'bitWidth' bit-width from the given +//! unsigned long long integer value. +//! +//! The 'bitWidth' must be large enough to fully store the given value's bit representation. +//! +DLL_PUBLIC Expr vc_bvConstExprFromLL(VC vc, int bitWidth, + unsigned long long value); + +//! \brief Returns a bitvector with a bit-width of 32 from the given +//! unsigned integer value. +//! +DLL_PUBLIC Expr vc_bv32ConstExprFromInt(VC vc, unsigned int value); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR ARITHMETIC OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a bitvector expression representing the concatenation of the two +//! given bitvector expressions. +//! +//! This results in a bitvector with the bit-width of the bit-width sum +//! of its children. +//! +//! Example: Given bitvector 'a = 1101' and 'b = 1000' then 'vc_bvConcatExpr(vc, a, b)' +//! results in 'c = 11011000'. +//! +DLL_PUBLIC Expr vc_bvConcatExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression representing the addition of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvPlusExpr(VC vc, int bitWidth, Expr left, Expr right); + +//! \brief Returns a bitvector expression representing the addition of the N +//! given bitvector expressions in the 'children' array. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvPlusExprN(VC vc, int bitWidth, Expr* children, + int numOfChildNodes); + +//! \brief Returns a bitvector expression with a bit-width of 32 +//! representing the addition of the two given bitvector expressions. +//! +//! The given bitvector expressions must have a bit-width of 32. +//! +DLL_PUBLIC Expr vc_bv32PlusExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the subtraction '(left - right)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvMinusExpr(VC vc, int bitWidth, Expr left, Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 32 +//! representing the subtraction '(left - right)' of the given +//! bitvector expressions. +//! +//! The given bitvector expressions must have a bit-width of 32. +//! +DLL_PUBLIC Expr vc_bv32MinusExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the multiplication '(left * right)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvMultExpr(VC vc, int bitWidth, Expr left, Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 32 +//! representing the multiplication '(left * right)' of the given +//! bitvector expressions. +//! +//! The given bitvector expressions must have a bit-width of 32. +//! +DLL_PUBLIC Expr vc_bv32MultExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the division '(dividend / divisor)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvDivExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the modulo '(dividend % divisor)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_bvModExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +//! \brief Returns a (signed) bitvector expression with a bit-width of 'bitWidth' +//! representing the signed division '(dividend / divisor)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_sbvDivExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +//! \brief Returns a (signed) bitvector expression with a bit-width of 'bitWidth' +//! representing the signed modulo '(dividend % divisor)' of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_sbvModExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +//! \brief Returns a (signed) bitvector expression with a bit-width of 'bitWidth' +//! representing the signed remainder of the two +//! given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width as 'bitWidth' +//! +DLL_PUBLIC Expr vc_sbvRemExpr(VC vc, int bitWidth, Expr dividend, Expr divisor); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR COMPARISON OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a boolean expression representing the less-than +//! operation '(left < right)' of the given bitvector expressions. +//! +DLL_PUBLIC Expr vc_bvLtExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the less-equals +//! operation '(left <= right)' of the given bitvector expressions. +//! +DLL_PUBLIC Expr vc_bvLeExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the greater-than +//! operation '(left > right)' of the given bitvector expressions. +//! +DLL_PUBLIC Expr vc_bvGtExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the greater-equals +//! operation '(left >= right)' of the given bitvector expressions. +//! +DLL_PUBLIC Expr vc_bvGeExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the signed less-than +//! operation '(left < right)' of the given signed bitvector expressions. +//! +DLL_PUBLIC Expr vc_sbvLtExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the signed less-equals +//! operation '(left <= right)' of the given signed bitvector expressions. +//! +DLL_PUBLIC Expr vc_sbvLeExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the signed greater-than +//! operation '(left > right)' of the given signed bitvector expressions. +//! +DLL_PUBLIC Expr vc_sbvGtExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a boolean expression representing the signed greater-equals +//! operation '(left >= right)' of the given signed bitvector expressions. +//! +DLL_PUBLIC Expr vc_sbvGeExpr(VC vc, Expr left, Expr right); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR BITWISE OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a bitvector expression representing the arithmetic +//! negation '(-a)' (unary minus) of the given child bitvector expression. +//! +DLL_PUBLIC Expr vc_bvUMinusExpr(VC vc, Expr child); + +//! \brief Returns a bitvector expression representing the bitwise-and +//! operation '(a & b)' for the given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width. +//! +DLL_PUBLIC Expr vc_bvAndExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression representing the bitwise-or +//! operation '(a | b)' for the given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width. +//! +DLL_PUBLIC Expr vc_bvOrExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression representing the bitwise-xor +//! operation '(a ^ b)' for the given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width. +//! +DLL_PUBLIC Expr vc_bvXorExpr(VC vc, Expr left, Expr right); + +//! \brief Returns a bitvector expression representing the bitwise-not +//! operation '~a' for the given bitvector expressions. +//! +//! The given bitvector expressions must have the same bit-width. +//! +DLL_PUBLIC Expr vc_bvNotExpr(VC vc, Expr child); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR SHIFT OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the left-shift operation '(left >> right)' of the +//! given bitvector expressions. +//! +//! Note: This is the new API for this kind of operation! +//! +DLL_PUBLIC Expr vc_bvLeftShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the right-shift operation '(left << right)' of the +//! given bitvector expressions. +//! +//! Note: This is the new API for this kind of operation! +//! +DLL_PUBLIC Expr vc_bvRightShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +//! \brief Returns a bitvector expression with a bit-width of 'bitWidth' +//! representing the signed right-shift operation '(left >> right)' of the +//! given bitvector expressions. +//! +//! Note: This is the new API for this kind of operation! +//! +DLL_PUBLIC Expr vc_bvSignedRightShiftExprExpr(VC vc, int bitWidth, Expr left, + Expr right); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns an expression representing the left-shift operation '(child << sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvLeftShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bvLeftShiftExpr(VC vc, int sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns an expression representing the right-shift operation '(child >> sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvRightShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bvRightShiftExpr(VC vc, int sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns a bitvector expression with a bit-width of 32 +//! representing the left-shift operation '(child << sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvLeftShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bv32LeftShiftExpr(VC vc, int sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns a bitvector expression with a bit-width of 32 +//! representing the right-shift operation '(child >> sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvRightShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bv32RightShiftExpr(VC vc, int sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns a bitvector expression with a bit-width of 32 +//! representing the left-shift operation '(child << sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvLeftShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bvVar32LeftShiftExpr(VC vc, Expr sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns a bitvector expression with a bit-width of 32 +//! representing the right-shift operation '(child >> sh_amt)' +//! for the given child bitvector expression. +//! +//! Note: Use 'vc_bvRightShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bvVar32RightShiftExpr(VC vc, Expr sh_amt, Expr child); + +//! \brief Deprecated: Use the new API instead! +//! +//! Returns a bitvector expression representing the division +//! operation of the power of two '(child / 2^rhs)' for the given +//! bitvector expressions. +//! +//! Note: Use 'vc_bvSignedRightShiftExprExpr' instead! +//! +DLL_PUBLIC Expr vc_bvVar32DivByPowOfTwoExpr(VC vc, Expr child, Expr rhs); + +///////////////////////////////////////////////////////////////////////////// +/// BITVECTOR EXTRACTION & EXTENSION +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a bitvector expression representing the extraction +//! of the bits within the range of 'low_bit_no' and 'high_bit_no'. +//! +//! Note: The resulting bitvector expression has a bit-width of 'high_bit_no - low_bit_no'. +//! +DLL_PUBLIC Expr vc_bvExtract(VC vc, Expr child, int high_bit_no, + int low_bit_no); + +//! \brief Superseeded: Use 'vc_bvBoolExtract_Zero' or 'vc_bvBoolExtract_One' instead. +//! +//! Returns a boolean expression that accepts a bitvector expression 'x' +//! and represents the following equation: '(x[bit_no:bit_no] == 0)'. +//! +//! Note: This is equal to calling 'vc_bvBoolExtract_Zero'. +//! +DLL_PUBLIC Expr vc_bvBoolExtract(VC vc, Expr x, int bit_no); + +//! \brief Returns a boolean expression that accepts a bitvector expression 'x' +//! and represents the following equation: '(x[bit_no:bit_no] == 0)'. +//! +DLL_PUBLIC Expr vc_bvBoolExtract_Zero(VC vc, Expr x, int bit_no); + +//! \brief Returns a boolean expression that accepts a bitvector expression 'x' +//! and represents the following equation: '(x[bit_no:bit_no] == 1)'. +//! +DLL_PUBLIC Expr vc_bvBoolExtract_One(VC vc, Expr x, int bit_no); + +//! \brief Returns a bitvector expression representing the extension of the given +//! to the amount of bits given by 'newWidth'. +//! +//! Note: This operation retains the signedness of the bitvector is existant. +//! +DLL_PUBLIC Expr vc_bvSignExtend(VC vc, Expr child, int newWidth); + +///////////////////////////////////////////////////////////////////////////// +/// CONVENIENCE FUNCTIONS FOR ARRAYS +///////////////////////////////////////////////////////////////////////////// + +/*C pointer support: C interface to support C memory arrays in CVCL */ + +//! \brief Convenience function to create a named array expression with +//! an index bit-width of 32 and a value bit-width of 8. +//! +DLL_PUBLIC Expr vc_bvCreateMemoryArray(VC vc, const char* arrayName); + +//! \brief Convenience function to read a bitvector with byte-width 'numOfBytes' of an +//! array expression created by 'vc_bvCreateMemoryArray' and return it. +//! +//! Note: This returns a bitvector expression with a bit-width of 'numOfBytes'. +//! +DLL_PUBLIC Expr vc_bvReadMemoryArray(VC vc, Expr array, Expr byteIndex, + int numOfBytes); + +//! \brief Convenience function to write a bitvector 'element' with byte-width 'numOfBytes' +//! into the given array expression at offset 'byteIndex'. +//! +DLL_PUBLIC Expr vc_bvWriteToMemoryArray(VC vc, Expr array, Expr byteIndex, + Expr element, int numOfBytes); + +///////////////////////////////////////////////////////////////////////////// +/// GENERAL EXPRESSION OPERATIONS +///////////////////////////////////////////////////////////////////////////// + +//! \brief Returns a string representation of the given expression. +//! +//! Note: +//! The caller is responsible for deallocating the string afterwards. +//! The buffer that stores the string is allocated by STP. +//! +DLL_PUBLIC char* exprString(Expr e); + +//! \brief Returns a string representation of the given type. +//! +//! Note: +//! The caller is responsible for deallocating the string afterwards. +//! The buffer that stores the string is allocated by STP. +//! +DLL_PUBLIC char* typeString(Type t); + +//! \brief Returns the n-th child of the given expression. +//! +DLL_PUBLIC Expr getChild(Expr e, int n); + +//! \brief Misleading name! +//! +//! Returns '1' if the given boolean expression evaluates to 'true', +//! returns '0' if the given boolean expression evaluates to 'false', +//! or returns '-1' otherwise, i.e. if the given expression was not a +//! boolean expression. +//! +DLL_PUBLIC int vc_isBool(Expr e); + +//! \brief Registers the given error handler function to be called for each +//! fatal error that occures while running STP. +//! +DLL_PUBLIC void +vc_registerErrorHandler(void (*error_hdlr)(const char* err_msg)); + +//! \brief Returns the hash of the given query state. +//! +DLL_PUBLIC int vc_getHashQueryStateToBuffer(VC vc, Expr query); + +//! \brief Destroy the given validity checker. +//! +//! Removes all associated expressions with it if 'EXPRDELETE' was set to 'true' +//! via 'vc_setInterfaceFlags' during the process. +//! +DLL_PUBLIC void vc_Destroy(VC vc); + +//! \brief Destroy the given expression, freeing its associated memory. +//! +DLL_PUBLIC void vc_DeleteExpr(Expr e); + +//! \brief Returns the whole counterexample from the given validity checker. +//! +DLL_PUBLIC WholeCounterExample vc_getWholeCounterExample(VC vc); + +//! \brief Returns the value of the given term expression from the given whole counter example. +//! +DLL_PUBLIC Expr vc_getTermFromCounterExample(VC vc, Expr e, + WholeCounterExample c); + +//! \brief Destroys the given whole counter example, freeing all of its associated memory. +//! +DLL_PUBLIC void vc_deleteWholeCounterExample(WholeCounterExample cc); + +//! Covers all kinds of expressions that exist in STP. +//! +enum exprkind_t +{ + UNDEFINED, //!< An undefined expression. + SYMBOL, //!< Named expression (or variable), i.e. created via 'vc_varExpr'. + BVCONST, //!< Bitvector constant expression, i.e. created via 'vc_bvConstExprFromInt'. + BVNOT, //!< Bitvector bitwise-not + BVCONCAT, //!< Bitvector concatenation + BVOR, //!< Bitvector bitwise-or + BVAND, //!< Bitvector bitwise-and + BVXOR, //!< Bitvector bitwise-xor + BVNAND, //!< Bitvector bitwise not-and; OR nand (TODO: does this still exist?) + BVNOR, //!< Bitvector bitwise not-or; OR nor (TODO: does this still exist?) + BVXNOR, //!< Bitvector bitwise not-xor; OR xnor (TODO: does this still exist?) + BVEXTRACT, //!< Bitvector extraction, i.e. via 'vc_bvExtract'. + BVLEFTSHIFT, //!< Bitvector left-shift + BVRIGHTSHIFT, //!< Bitvector right-right + BVSRSHIFT, //!< Bitvector signed right-shift + BVPLUS, //!< Bitvector addition + BVSUB, //!< Bitvector subtraction + BVUMINUS, //!< Bitvector unary minus; OR negate expression + BVMULT, //!< Bitvector multiplication + BVDIV, //!< Bitvector division + BVMOD, //!< Bitvector modulo operation + SBVDIV, //!< Signed bitvector division + SBVREM, //!< Signed bitvector remainder + SBVMOD, //!< Signed bitvector modulo operation + BVSX, //!< Bitvector signed extend + BVZX, //!< Bitvector zero extend + ITE, //!< If-then-else + BOOLEXTRACT, //!< Bitvector boolean extraction + BVLT, //!< Bitvector less-than + BVLE, //!< Bitvector less-equals + BVGT, //!< Bitvector greater-than + BVGE, //!< Bitvector greater-equals + BVSLT, //!< Signed bitvector less-than + BVSLE, //!< Signed bitvector less-equals + BVSGT, //!< Signed bitvector greater-than + BVSGE, //!< Signed bitvector greater-equals + EQ, //!< Equality comparator + FALSE, //!< Constant false boolean expression + TRUE, //!< Constant true boolean expression + NOT, //!< Logical-not boolean expression + AND, //!< Logical-and boolean expression + OR, //!< Logical-or boolean expression + NAND, //!< Logical-not-and boolean expression (TODO: Does this still exist?) + NOR, //!< Logical-not-or boolean expression (TODO: Does this still exist?) + XOR, //!< Logical-xor (either-or) boolean expression + IFF, //!< If-and-only-if boolean expression + IMPLIES, //!< Implication boolean expression + PARAMBOOL, //!< Parameterized boolean expression + READ, //!< Array read expression + WRITE, //!< Array write expression + ARRAY, //!< Array creation expression + BITVECTOR, //!< Bitvector creation expression + BOOLEAN //!< Boolean creation expression +}; + +//! \brief Returns the expression-kind of the given expression. +//! +DLL_PUBLIC enum exprkind_t getExprKind(Expr e); + +//! \brief Returns the number of child expressions of the given expression. +//! +DLL_PUBLIC int getDegree(Expr e); + +//! \brief Returns the bit-width of the given bitvector expression. +//! +DLL_PUBLIC int getBVLength(Expr e); + +//! Covers all kinds of types that exist in STP. +//! +enum type_t +{ + BOOLEAN_TYPE = 0, + BITVECTOR_TYPE, + ARRAY_TYPE, + UNKNOWN_TYPE +}; + +//! \brief Returns the type-kind of the given expression. +//! +DLL_PUBLIC enum type_t getType(Expr e); + +// get value bit width + +//! \brief Returns the value bit-width of the given expression. +//! +//! This is mainly useful for array expression. +//! +DLL_PUBLIC int getVWidth(Expr e); + +//! \brief Returns the index bit-width of the given expression. +//! +//! This is mainly useful for array expression. +//! +DLL_PUBLIC int getIWidth(Expr e); + +//! \brief Prints the given counter example to the file that is +//! associated with the given open file descriptor. +//! +DLL_PUBLIC void vc_printCounterExampleFile(VC vc, int fd); + +//! \brief Returns the name of the given variable expression. +//! +DLL_PUBLIC const char* exprName(Expr e); + +//! \brief Returns the internal node ID of the given expression. +//! +DLL_PUBLIC int getExprID(Expr ex); + +//! \brief Parses the given string in CVC or SMTLib1.0 format and extracts +//! query and assertion information into the 'outQuery' and 'outAsserts' +//! buffers respectively. +//! +//! It is the caller's responsibility to free the buffer's memory afterwards. +//! +//! Note: The user can controle the parsed format via 'process_argument'. +//! +//! Returns '1' if parsing was successful. +//! +DLL_PUBLIC int vc_parseMemExpr(VC vc, const char* s, Expr* outQuery, + Expr* outAsserts); + +//! \brief Checks if STP was compiled with support for minisat +//! +//! Note: always returns true (future support for minisat being the +//! non-default) +//! +DLL_PUBLIC bool vc_supportsMinisat(VC vc); + +//! \brief Sets underlying SAT solver to minisat +//! +DLL_PUBLIC bool vc_useMinisat(VC vc); + +//! \brief Checks if underlying SAT solver is minisat +//! +DLL_PUBLIC bool vc_isUsingMinisat(VC vc); + +//! \brief Checks if STP was compiled with support for simplifying minisat +//! +//! Note: always returns true (future support for simplifying minisat being +//! the non-default) +//! +DLL_PUBLIC bool vc_supportsSimplifyingMinisat(VC vc); + +//! \brief Sets underlying SAT solver to simplifying minisat +//! +DLL_PUBLIC bool vc_useSimplifyingMinisat(VC vc); + +//! \brief Checks if underlying SAT solver is simplifying minisat +//! +DLL_PUBLIC bool vc_isUsingSimplifyingMinisat(VC vc); + +//! \brief Checks if STP was compiled with support for cryptominisat +//! +DLL_PUBLIC bool vc_supportsCryptominisat(VC vc); + +//! \brief Sets underlying SAT solver to cryptominisat +//! +DLL_PUBLIC bool vc_useCryptominisat(VC vc); + +//! \brief Checks if underlying SAT solver is cryptominisat +//! +DLL_PUBLIC bool vc_isUsingCryptominisat(VC vc); + +//! \brief Checks if STP was compiled with support for riss +//! +DLL_PUBLIC bool vc_supportsRiss(VC vc); + +//! \brief Sets underlying SAT solver to riss +//! +DLL_PUBLIC bool vc_useRiss(VC vc); + +//! \brief Checks if underlying SAT solver is riss +//! +DLL_PUBLIC bool vc_isUsingRiss(VC vc); + + +#ifdef __cplusplus +} +#endif + +#undef DLL_PUBLIC // Undefine internal macro to prevent it from leaking into the API. +#undef DLL_LOCAL // Undefine internal macro to prevent it from leaking into the API. + +#undef _CVCL_DEFAULT_ARG // Undefine macro to not pollute global macro namespace! + +#endif // _cvcl__include__c_interface_h_ diff --git a/native-library-files/stp_project/stp_extend/ext_c_interface.cpp b/native-library-files/stp_project/stp_extend/ext_c_interface.cpp new file mode 100644 index 0000000000..7f76fc34a3 --- /dev/null +++ b/native-library-files/stp_project/stp_extend/ext_c_interface.cpp @@ -0,0 +1,178 @@ +#include "stp/c_interface.h" + +#include +#include +#include + +#include "stp/Interface/fdstream.h" +#include "stp/Parser/parser.h" +#include "stp/Printer/printers.h" +#include "stp/cpp_interface.h" +#include "stp/Util/GitSHA1.h" +// FIXME: External library +#include "extlib-abc/cnf_short.h" + +using std::cout; +using std::ostream; +using std::stringstream; +using std::string; +using std::fdostream; +using std::endl; + + +// Add a new formula (assertion or query) in the current context. +/*! The formula must have Boolean type. */ +void addAssertFormula(VC vc, Expr e) +{ + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* b = stp_i->bm; + stp::ASTNode* a = (stp::ASTNode*)e; + + stp::Cpp_interface cpp_inter(*b, b->defaultNodeFactory); + cpp_inter.AddAssert(*a); + + /* + if (!stp::is_Form_kind(a->GetKind())) + stp::FatalError("Trying to assert a NON formula: ", *a); + + assert(BVTypeCheck(*a)); + b->AddAssert(*a); + */ +} + +void push(VC vc) +{ + + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* b = stp_i->bm; + + stp_i->ClearAllTables(); + b->Push(); + +} + +void pop(VC vc) +{ + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* b = stp_i->bm; + + b->Pop(); +} + +int checkSAT_old(VC vc) +{ + // return vc_query_with_timeout(vc, vc_falseExpr(vc), -1, -1); + return vc_query_with_timeout(vc, vc_trueExpr(vc), -1, -1); +} + + +int checkSAT(VC vc, Expr e) +{ + // stp::STP* stp_i = (stp::STP*)vc; + // stp::STPMgr* b = stp_i->bm; + + // stp::Cpp_interface cpp_inter(*b, b->defaultNodeFactory); + + // const stp::ASTVec asserts = cpp_inter.GetAsserts(); + // cpp_inter.checkSat(asserts); + + return 2; //ERROR + +} + +void getSTPenv(VC vc) +{ + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* b = stp_i->bm; + + stp::Cpp_interface cpp_inter(*b, b->defaultNodeFactory); + + // const stp::ASTVec asserts = cpp_inter.GetAsserts(); + // cpp_inter.checkSat(asserts); + +} + +// extern "C" +// { + StpEnv * createStpEnv(VC vc) + { + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* b = stp_i->bm; + return reinterpret_cast(new stp::Cpp_interface(*b, b->defaultNodeFactory)); + } + + void destroyStpEnv(StpEnv * env) + { + delete reinterpret_cast(env); + } + + int getCacheSize(StpEnv * env) + { + return reinterpret_cast(env)->getCacheSize(); + } + + int getSymbolsSize(StpEnv * env) + { + return reinterpret_cast(env)->getSymbolsSize(); + } + + void ext_push(StpEnv * env) + { + reinterpret_cast(env)->push(); + } + + void ext_pop(StpEnv * env) + { + reinterpret_cast(env)->pop(); + } + + void ext_addFormula(StpEnv * env, Expr formula) + { + stp::ASTNode* assertn = (stp::ASTNode*)formula; + + // TODO: handle error properly; Are these checks even needed !!! + // if (!stp::is_Form_kind(assertn->GetKind())) + // stp::FatalError("Trying to assert a NON formula: ", *assertn); + + // assert(stp::BVTypeCheck(*assertn)); + + reinterpret_cast(env)->AddAssert(*assertn); + } + + void ext_checkSat(StpEnv * env) + { + auto ptr = reinterpret_cast(env); + auto assertions = ptr->getAssertVector(); + cout<<"***Number of current assertion is : "<< assertions.size()<<"\n"; + ptr->checkSat(assertions); + } + + const char * getAllModel(VC vc) + { + stp::STP* stp_i = (stp::STP*)vc; + stp::STPMgr* bm = stp_i->bm; + + // if (!bm->UserFlags.construct_counterexample_flag) + // { + // // unsupported + // return; + // } + + // if (cache.size() ==0 || (cache.back().result != SOLVER_SATISFIABLE)) + // { + // return; + // } + + stp::AbsRefine_CounterExample* Ctr_Example = + (stp::AbsRefine_CounterExample*)(stp_i->Ctr_Example); + + std::ostringstream outputStream; + outputStream << "(model" << std::endl; + Ctr_Example->PrintFullCounterExampleSMTLIB2(outputStream); + outputStream << ")" << std::endl; + + std::string returnStr = outputStream.str(); + return returnStr.c_str(); + } + +// } \ No newline at end of file diff --git a/native-library-files/testproject/py_network/send_obj/notes.md b/native-library-files/testproject/py_network/send_obj/notes.md new file mode 100644 index 0000000000..b1726a1245 --- /dev/null +++ b/native-library-files/testproject/py_network/send_obj/notes.md @@ -0,0 +1,18 @@ +# NOTES + +## import pickle + +Pickel is used to serialise python objects + +https://www.youtube.com/playlist?list=PLQVvvaa0QuDdzLB_0JSTTcl8E8jsJLhR5 + +Websockets: + +https://www.youtube.com/watch?v=PjiXkJ6P9pQ&t=28s + +https://www.youtube.com/watch?v=9FqjRN4VYUU + +https://www.youtube.com/watch?v=i5OVcTdt_OU + +https://www.youtube.com/watch?v=zgI0H28AgGY + diff --git a/src/org/sosy_lab/java_smt/SolverContextFactory.java b/src/org/sosy_lab/java_smt/SolverContextFactory.java index 93cc9006b6..2a9af7f103 100644 --- a/src/org/sosy_lab/java_smt/SolverContextFactory.java +++ b/src/org/sosy_lab/java_smt/SolverContextFactory.java @@ -54,6 +54,7 @@ import org.sosy_lab.java_smt.solvers.mathsat5.Mathsat5SolverContext; import org.sosy_lab.java_smt.solvers.princess.PrincessSolverContext; import org.sosy_lab.java_smt.solvers.smtinterpol.SmtInterpolSolverContext; +import org.sosy_lab.java_smt.solvers.stp.StpSolverContext; /** * Factory class for loading and generating solver contexts. Generates a {@link SolverContext} @@ -68,7 +69,8 @@ public enum Solvers { MATHSAT5, SMTINTERPOL, Z3, - PRINCESS + PRINCESS, + STP } @Option(secure = true, description = "Export solver queries in SmtLib format into a file.") @@ -214,6 +216,8 @@ private SolverContext generateContext0(Solvers solverToCreate) case PRINCESS: return PrincessSolverContext.create( config, shutdownNotifier, logfile, (int) randomSeed, nonLinearArithmetic); + case STP: + return StpSolverContext.create(config, logger, shutdownNotifier, logfile, randomSeed); default: throw new AssertionError("no solver selected"); diff --git a/src/org/sosy_lab/java_smt/example/BasicSample1.java b/src/org/sosy_lab/java_smt/example/BasicSample1.java new file mode 100644 index 0000000000..5219f6ca25 --- /dev/null +++ b/src/org/sosy_lab/java_smt/example/BasicSample1.java @@ -0,0 +1,197 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.example; + +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.common.configuration.Configuration; +import org.sosy_lab.common.configuration.InvalidConfigurationException; +import org.sosy_lab.common.log.BasicLogManager; +import org.sosy_lab.common.log.LogManager; +import org.sosy_lab.java_smt.SolverContextFactory; +import org.sosy_lab.java_smt.SolverContextFactory.Solvers; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.BooleanFormulaManager; +import org.sosy_lab.java_smt.api.IntegerFormulaManager; +import org.sosy_lab.java_smt.api.NumeralFormula.IntegerFormula; +import org.sosy_lab.java_smt.api.ProverEnvironment; +import org.sosy_lab.java_smt.api.SolverContext; +import org.sosy_lab.java_smt.api.SolverContext.ProverOptions; +import org.sosy_lab.java_smt.api.SolverException; + +public class BasicSample1 { + private BasicSample1() {} + + public static void main(String[] args) + throws InvalidConfigurationException, SolverException, InterruptedException { + + // Setup and Congigurations + Configuration config = Configuration.fromCmdLineArguments(args); + LogManager logger = BasicLogManager.create(config); + + // deal with cleanup and shutdowns + // ShutdownManager shutdown = ShutdownManager.create(); + ShutdownNotifier shutdownNotifier = ShutdownNotifier.createDummy(); + + // We can do this too but ... + // SolverContext context = SolverContextFactory.createSolverContext( + // config, logger, shutdown.getNotifier(), Solvers.SMTINTERPOL); + + // Solvers solver = Solvers.SMTINTERPOL; + // + for (Solvers solver : Solvers.values()) { + System.out.println("NOW TESTING WITH : " + solver + ".. \n"); + + // Instantiate JavaSMT with SMTInterpol as backend (for dependencies cf. documentation) + try (SolverContext context = + SolverContextFactory.createSolverContext(config, logger, shutdownNotifier, solver)) { + + /* BOOLEAN THEORY */ + // create the manager + BooleanFormulaManager booleFormularMgr = + context.getFormulaManager().getBooleanFormulaManager(); + + // create atoms + BooleanFormula xL = booleFormularMgr.makeVariable("xL"); + BooleanFormula xH = booleFormularMgr.makeVariable("xH"); + BooleanFormula yL = booleFormularMgr.makeVariable("yL"); + BooleanFormula yH = booleFormularMgr.makeVariable("yH"); + + // create formular + BooleanFormula lowXOR = booleFormularMgr.xor(xL, yL); + BooleanFormula highXOR = booleFormularMgr.xor(xH, yH); + BooleanFormula twoBitAdder = booleFormularMgr.and(lowXOR, highXOR); // Formula to solve + + /* LRA (Integer) THEORY */ + // create the manager + IntegerFormulaManager intFormularMgr = + context.getFormulaManager().getIntegerFormulaManager(); + + // create atoms + IntegerFormula x = intFormularMgr.makeVariable("x"); + IntegerFormula y = intFormularMgr.makeVariable("y"); + + // create formula + + IntegerFormula twoX = intFormularMgr.multiply(x, intFormularMgr.makeNumber(2)); + IntegerFormula threeX = intFormularMgr.multiply(x, intFormularMgr.makeNumber(3)); + IntegerFormula twoY = intFormularMgr.multiply(y, intFormularMgr.makeNumber(2)); + + // 3*x + y = 11 + BooleanFormula eqA1 = + intFormularMgr.equal(intFormularMgr.add(threeX, y), intFormularMgr.makeNumber(11)); + // 2*x + y = 8 + BooleanFormula eqA2 = + intFormularMgr.equal(intFormularMgr.add(twoX, y), intFormularMgr.makeNumber(8)); + + // Formula to solve (3,2) + BooleanFormula intTheorySampleA = booleFormularMgr.and(eqA1, eqA2); + + // x>2 + BooleanFormula eqB1 = intFormularMgr.greaterThan(x, intFormularMgr.makeNumber(2)); + // y<10 + BooleanFormula eqB2 = intFormularMgr.lessThan(y, intFormularMgr.makeNumber(10)); + // x+2*y == 7 (how do I differentiate) + BooleanFormula eqB3 = + intFormularMgr.equal(intFormularMgr.add(x, twoY), intFormularMgr.makeNumber(7)); + + // Formula to solve (0,7) + BooleanFormula intTheorySampleB = booleFormularMgr.and(eqB1, eqB2, eqB3); + + /* LRA (Rational) THEORY */ + // create the manager + // RationalFormulaManager rationalFormularMgr = + // context.getFormulaManager().getRationalFormulaManager(); + + // create atoms + // RationalFormula a = rationalFormularMgr.makeVariable("a"); + // RationalFormula b = rationalFormularMgr.makeVariable("b"); + + // RationalFormula a1 = rationalFormularMgr.makeVariable("a1"); + // RationalFormula a2 = rationalFormularMgr.makeVariable("a2"); + // RationalFormula b1 = rationalFormularMgr.makeVariable("b1"); + // RationalFormula b2 = rationalFormularMgr.makeVariable("b2"); + + // create formula + /* + * UNSUPPORTED a^2 ==> this is linear + * + * RationalFormula a_square = rationalFormularMgr.multiply(a, a); RationalFormula b_square = + * rationalFormularMgr.multiply(b, b); // a^2 + b^2 < 1 BooleanFormula eqR1 = + * rationalFormularMgr.lessThan( rationalFormularMgr.add(a_square, b_square), + * rationalFormularMgr.makeNumber(1)); // x*y > 0.1 BooleanFormula eqR2 = rationalFormularMgr + * .greaterThan(rationalFormularMgr.multiply(a, b), rationalFormularMgr.makeNumber(0.1)); + * + * BooleanFormula ratTheorySample1 = booleFormularMgr.and(eqR1, eqR2); // Formula to solve + * (1/8, // 7/8) + * + * // x*y > 1 BooleanFormula eqR3 = rationalFormularMgr + * .greaterThan(rationalFormularMgr.multiply(a, b), rationalFormularMgr.makeNumber(1)); + * + * BooleanFormula ratTheorySample2 = booleFormularMgr.and(eqR1, eqR3); // Formula to solve // + * (UNSAT) + */ + + boolean isUnsat; + + // Solve formulae, get model, and print variable assignment + try (ProverEnvironment prover = + context.newProverEnvironment(ProverOptions.GENERATE_MODELS)) { + + prover.addConstraint(twoBitAdder); + isUnsat = prover.isUnsat(); + assert !isUnsat; + // try (Model model = prover.getModel()) { + System.out.println("SAT : 2-bit Adder "); + // } + + prover.addConstraint(intTheorySampleA); + isUnsat = prover.isUnsat(); + assert !isUnsat; + // try (Model model = prover.getModel()) { + // System.out.printf("SAT with a = %s, b = %s", model.evaluate(a), model.evaluate(b)); + System.out.println("SAT : 2 equations 3*x + y = 11 AND 2*x + y = 8"); + // } + + prover.addConstraint(intTheorySampleB); + isUnsat = prover.isUnsat(); + assert !isUnsat; + // try (Model model = prover.getModel()) { + System.out.println("SAT : 3 equations x>2 AND y<10 AND x+2*y == 7"); + // } + + // prover.addConstraint(ratTheorySample1); + // isUnsat = prover.isUnsat(); + // assert !isUnsat; + // // try (Model model = prover.getModel()) { + // System.out.println("SAT : 2 equations a^2 + b^2 < 1 AND x*y > 0.1 "); + // // } + // + // prover.addConstraint(ratTheorySample2); + // isUnsat = prover.isUnsat(); + // assert isUnsat; + // // try (Model model = prover.getModel()) { + // System.out.println("UNSAT : 2 equations a^2 + b^2 < 1 AND x*y > 1 "); + // // } + + } + } + } + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpAbstractProver.java b/src/org/sosy_lab/java_smt/solvers/stp/StpAbstractProver.java new file mode 100644 index 0000000000..8be9ab5c96 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpAbstractProver.java @@ -0,0 +1,137 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import com.google.common.base.Preconditions; +import java.util.Collection; +import java.util.List; +import java.util.Optional; +import java.util.Set; +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.Model; +import org.sosy_lab.java_smt.api.SolverContext.ProverOptions; +import org.sosy_lab.java_smt.api.SolverException; +import org.sosy_lab.java_smt.basicimpl.AbstractProver; + +abstract class StpAbstractProver extends AbstractProver { + + private final StpSolverContext context; + private final StpFormulaCreator creator; + private final ShutdownNotifier shutdownNotifier; + // private final long curConfig; + protected final VC currVC; + protected boolean closed; + + protected StpAbstractProver( + StpSolverContext pContext, + Set pOptions, + StpFormulaCreator pCreator, + ShutdownNotifier pShutdownNotifier) { + super(pOptions); + context = pContext; + creator = pCreator; + // curConfig = buildConfig(pOptions); //TODO implement configuration handling + currVC = context.createEnvironment(null); // curConfig is to be passed in here + shutdownNotifier = pShutdownNotifier; + } + + @Override + public void pop() { + Preconditions.checkState(!closed); + StpJavaApi.vc_pop(currVC); + } + + @Override + public boolean isUnsat() throws SolverException, InterruptedException { + // TODO update to use vc_query_with_timeout + + // To go this route I will have to implement the Stack for the "Constraints" ?! + + Preconditions.checkState(!closed); + int result = StpJavaApi.checkSAT_old(currVC); + try { + if (result == 0) { + return true; + } else if (result == 1) { + return false; + } else if (result == 2) { + throw new Exception("An error occured in STP during validation"); + } + + throw new Exception("An error occured in STP during validation"); + } catch (Exception e) { + // TODO Auto-generated catch block + e.printStackTrace(); + } + return false; + + // throw new SolverException("NOT MPLEMENTED"); + } + + @Override + public boolean isUnsatWithAssumptions(Collection pAssumptions) + throws SolverException, InterruptedException { + // TODO Auto-generated method stub + // return false; + throw new UnsupportedOperationException("Not implemented yet"); + } + + @Override + public R allSat(AllSatCallback pCallback, List pImportant) + throws InterruptedException, SolverException { + // TODO Auto-generated method stub + return null; + } + + @Override + public List getUnsatCore() { + // TODO Auto-generated method stub + return null; + } + + @Override + public Optional> unsatCoreOverAssumptions( + Collection pAssumptions) throws SolverException, InterruptedException { + // TODO Auto-generated method stub + return null; + } + + @Override + public Model getModel() throws SolverException { + // TODO Auto-generated method stub + + // I don't understand this model stuff. + // return null; + + throw new UnsupportedOperationException("Not yet implemented"); + } + + @Override + public void close() { + if (!closed) { + + // TODO check if EXPRDELETE is set via vc_setInterfaceFlags + // other we will can delete expression with vc_DeleteExpr + StpJavaApi.vc_Destroy(currVC); + closed = true; + } + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpArrayFormulaManager.java b/src/org/sosy_lab/java_smt/solvers/stp/StpArrayFormulaManager.java new file mode 100644 index 0000000000..676e5a1975 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpArrayFormulaManager.java @@ -0,0 +1,74 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static com.google.common.base.Preconditions.checkArgument; + +import org.sosy_lab.java_smt.api.Formula; +import org.sosy_lab.java_smt.api.FormulaType; +import org.sosy_lab.java_smt.api.FormulaType.ArrayFormulaType; +import org.sosy_lab.java_smt.basicimpl.AbstractArrayFormulaManager; + +class StpArrayFormulaManager extends AbstractArrayFormulaManager { + + private final VC vc; + + public StpArrayFormulaManager(StpFormulaCreator pFormulaCreator) { + super(pFormulaCreator); + this.vc = pFormulaCreator.getEnv(); + } + + @Override + protected Expr select(Expr pArray, Expr pIndex) { + boolean checkArray = type_t.ARRAY_TYPE.equals(StpJavaApi.getType(pArray)); + checkArgument(checkArray, "Argument not of type ARRAY"); + boolean checkIndex = type_t.BITVECTOR_TYPE.equals(StpJavaApi.getType(pIndex)); + checkArgument(checkIndex, "Argument not of type BITVECTOR"); + + return StpJavaApi.vc_readExpr(vc, pArray, pIndex); + } + + @Override + protected Expr store(Expr pArray, Expr pIndex, Expr pValue) { + boolean checkArray = type_t.ARRAY_TYPE.equals(StpJavaApi.getType(pArray)); + checkArgument(checkArray, "Argument not of type ARRAY"); + boolean checkIndex = type_t.BITVECTOR_TYPE.equals(StpJavaApi.getType(pIndex)); + checkArgument(checkIndex, "Argument not of type BITVECTOR"); + boolean checkValue = type_t.BITVECTOR_TYPE.equals(StpJavaApi.getType(pValue)); + checkArgument(checkValue, "Argument not of type BITVECTOR"); + + return StpJavaApi.vc_writeExpr(vc, pArray, pIndex, pValue); + } + + @Override + protected Expr internalMakeArray( + String pName, FormulaType pIndexType, FormulaType pElementType) { + + ArrayFormulaType arrayFormulaType = FormulaType.getArrayType(pIndexType, pElementType); + Type stpArrayType = toSolverType(arrayFormulaType); + + return getFormulaCreator().makeVariable(stpArrayType, pName); + } + + @Override + protected Expr equivalence(Expr pArray1, Expr pArray2) { + return StpJavaApi.vc_eqExpr(vc, pArray1, pArray2); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpBitvectorFormulaManager.java b/src/org/sosy_lab/java_smt/solvers/stp/StpBitvectorFormulaManager.java new file mode 100644 index 0000000000..39dd8031a1 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpBitvectorFormulaManager.java @@ -0,0 +1,206 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static com.google.common.base.Preconditions.checkArgument; + +import java.math.BigInteger; +import org.sosy_lab.java_smt.basicimpl.AbstractBitvectorFormulaManager; + +class StpBitvectorFormulaManager extends AbstractBitvectorFormulaManager { + + private final VC vc; + + protected StpBitvectorFormulaManager(StpFormulaCreator pCreator) { + super(pCreator); + this.vc = pCreator.getEnv(); + } + + public static StpBitvectorFormulaManager create(StpFormulaCreator creator) { + return new StpBitvectorFormulaManager(creator); + } + + @Override + protected Expr makeVariableImpl(int pLength, String pVar) { + Type bvType = getFormulaCreator().getBitvectorType(pLength); + return getFormulaCreator().makeVariable(bvType, pVar); + } + + @Override + public Expr makeBitvectorImpl(int pLength, long pI) { + int i = (int) pI; + if (i == pI && i > 0) { // fits into an int + return StpJavaApi.vc_bv32ConstExprFromInt(vc, pI); // msat_make_bv_int_number(mathsatEnv, i, + // pLength); + } + return makeBitvectorImpl(pLength, BigInteger.valueOf(pI)); + } + + @Override + protected Expr makeBitvectorImpl(int pLength, BigInteger pI) { + if (pI.signum() < 0) { + BigInteger max = BigInteger.valueOf(2).pow(pLength - 1); + if (pI.compareTo(max.negate()) < 0) { + throw new IllegalArgumentException( + pI + " is to small for a bitvector with length " + pLength); + } + BigInteger n = BigInteger.valueOf(2).pow(pLength); + pI = pI.add(n); + } + return StpJavaApi.vc_bvConstExprFromLL(vc, pLength, pI); + } + + @Override + protected Expr makeBitvectorImpl(int pLength, Expr pParam1) { + // TODO Implement this + // return StpJavaApi.vc_bv32ConstExprFromInt(vc, pLength); + throw new UnsupportedOperationException(" not yet implemented yet"); + } + + @Override + protected Expr toIntegerFormulaImpl(Expr pI, boolean pSigned) { + // TODO IntegerFormula is not in STP + // return null; + throw new UnsupportedOperationException(" not yet implemented yet"); + } + + @Override + protected Expr negate(Expr pParam1) { + return StpJavaApi.vc_bvUMinusExpr(vc, pParam1); + } + + @Override + protected Expr add(Expr pParam1, Expr pParam2) { + int bitSize = confirmEqualBitSize(pParam1, pParam2); + return StpJavaApi.vc_bvPlusExpr(vc, bitSize, pParam1, pParam2); + } + + private int confirmEqualBitSize(Expr pParam1, Expr pParam2) { + int bitSize = StpJavaApi.getBVInt(pParam1); + checkArgument(bitSize == StpJavaApi.getBVInt(pParam2), "Formulae have different bit size"); + return bitSize; + } + + @Override + protected Expr subtract(Expr pParam1, Expr pParam2) { + int bitSize = confirmEqualBitSize(pParam1, pParam2); + return StpJavaApi.vc_bvMinusExpr(vc, bitSize, pParam1, pParam2); + } + + @Override + protected Expr divide(Expr pParam1, Expr pParam2, boolean pSigned) { + int bitSize = confirmEqualBitSize(pParam1, pParam2); + return StpJavaApi.vc_bvDivExpr(vc, bitSize, pParam1, pParam2); + } + + @Override + protected Expr modulo(Expr pParam1, Expr pParam2, boolean pSigned) { + int bitSize = confirmEqualBitSize(pParam1, pParam2); + return StpJavaApi.vc_bvModExpr(vc, bitSize, pParam1, pParam2); + } + + @Override + protected Expr multiply(Expr pParam1, Expr pParam2) { + int bitSize = confirmEqualBitSize(pParam1, pParam2); + return StpJavaApi.vc_bvModExpr(vc, bitSize, pParam1, pParam2); + } + + @Override + protected Expr equal(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_eqExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr greaterThan(Expr pParam1, Expr pParam2, boolean pSigned) { + return StpJavaApi.vc_bvGtExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr greaterOrEquals(Expr pParam1, Expr pParam2, boolean pSigned) { + return StpJavaApi.vc_bvGeExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr lessThan(Expr pParam1, Expr pParam2, boolean pSigned) { + return StpJavaApi.vc_bvLtExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr lessOrEquals(Expr pParam1, Expr pParam2, boolean pSigned) { + return StpJavaApi.vc_bvLeExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr not(Expr pParam1) { + return StpJavaApi.vc_bvNotExpr(vc, pParam1); + } + + @Override + protected Expr and(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_bvAndExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr or(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_bvOrExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr xor(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_bvXorExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr shiftRight(Expr pNumber, Expr pToShift, boolean pSigned) { + int bitSize = confirmEqualBitSize(pNumber, pToShift); + Expr retValue; + if (pSigned) { + retValue = StpJavaApi.vc_bvSignedRightShiftExprExpr(vc, bitSize, pNumber, pToShift); + } else { + retValue = StpJavaApi.vc_bvRightShiftExprExpr(vc, bitSize, pNumber, pToShift); + } + return retValue; + } + + @Override + protected Expr shiftLeft(Expr pExtract, Expr pExtract2) { + int bitSize = confirmEqualBitSize(pExtract, pExtract); + return StpJavaApi.vc_bvLeftShiftExprExpr(vc, bitSize, pExtract, pExtract); + } + + @Override + protected Expr concat(Expr pNumber, Expr pAppend) { + return StpJavaApi.vc_bvConcatExpr(vc, pNumber, pAppend); + } + + @Override + protected Expr extract(Expr pNumber, int pMsb, int pLsb, boolean pSigned) { + return StpJavaApi.vc_bvExtract(vc, pNumber, pMsb, pLsb); + } + + @Override + protected Expr extend(Expr pNumber, int pExtensionBits, boolean pSigned) { + if (pSigned) { + return StpJavaApi.vc_bvSignExtend(vc, pNumber, pExtensionBits); + } else { // TODO this should be using Unsigned Version + return StpJavaApi.vc_bvSignExtend(vc, pNumber, pExtensionBits); + } + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpBooleanFormulaManager.java b/src/org/sosy_lab/java_smt/solvers/stp/StpBooleanFormulaManager.java new file mode 100644 index 0000000000..c0a5860a07 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpBooleanFormulaManager.java @@ -0,0 +1,126 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static com.google.common.base.Preconditions.checkArgument; + +import org.sosy_lab.java_smt.basicimpl.AbstractBooleanFormulaManager; + +class StpBooleanFormulaManager extends AbstractBooleanFormulaManager { + + private final VC vc; + + protected StpBooleanFormulaManager(StpFormulaCreator pCreator) { + super(pCreator); + vc = pCreator.getEnv(); + } + + @Override + protected Expr makeVariableImpl(String pVar) { + Type boolType = getFormulaCreator().getBoolType(); + return getFormulaCreator().makeVariable(boolType, pVar); + } + + @Override + protected Expr makeBooleanImpl(boolean pValue) { + Expr result; + if (pValue) { + result = StpJavaApi.vc_trueExpr(vc); + } else { + result = StpJavaApi.vc_falseExpr(vc); + } + return result; + } + + @Override + protected Expr not(Expr pParam1) { + return StpJavaApi.vc_notExpr(vc, pParam1); + } + + @Override + protected Expr and(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_andExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr or(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_orExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr xor(Expr pParam1, Expr pParam2) { + return StpJavaApi.vc_xorExpr(vc, pParam1, pParam2); + } + + @Override + protected Expr equivalence(Expr pBits1, Expr pBits2) { + + boolean check = StpJavaApi.getType(pBits1).equals(StpJavaApi.getType(pBits2)); + checkArgument(check, "STP allows equivalence only for Formulae of the same type"); + + return StpJavaApi.vc_eqExpr(vc, pBits1, pBits2); + } + + @Override + protected boolean isTrue(Expr pBits) { + exprkind_t result = StpJavaApi.getExprKind(pBits); + switch (result) { + case TRUE: + return true; + case FALSE: + return false; + default: + throw new IllegalArgumentException( + "In STP solver: Formula of type - " + result + "needs to be SAT checked."); + } + } + + @Override + protected boolean isFalse(Expr pBits) { + + exprkind_t result = StpJavaApi.getExprKind(pBits); + switch (result) { + case TRUE: + return false; + case FALSE: + return true; + default: + throw new IllegalArgumentException( + "In STP solver: Formula of type - " + result + "needs to be SAT checked."); + } + } + + @Override + protected Expr ifThenElse(Expr cond, Expr thenExpr, Expr elseExpr) { + + boolean checkConditon = StpJavaApi.getType(cond).equals(type_t.BOOLEAN_TYPE); + checkArgument(checkConditon, "The conditon for If-Then-Else must be a Boolean type"); + + type_t typeThen = StpJavaApi.getType(thenExpr); + type_t typeElse = StpJavaApi.getType(elseExpr); + + checkArgument(typeThen.equals(typeElse), "Both Then and Else clauses must be of the same type"); + + boolean check = typeThen.equals(type_t.BITVECTOR_TYPE) || typeThen.equals(type_t.BOOLEAN_TYPE); + checkArgument(check, "Both Then and Else clauses must be either of BOOLEAN or BITVECTOR type"); + + return StpJavaApi.vc_iteExpr(vc, cond, thenExpr, elseExpr); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpEnvironment.java b/src/org/sosy_lab/java_smt/solvers/stp/StpEnvironment.java new file mode 100644 index 0000000000..0596de7bf6 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpEnvironment.java @@ -0,0 +1,62 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import org.checkerframework.checker.nullness.qual.Nullable; +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.common.configuration.Configuration; +import org.sosy_lab.common.io.PathCounterTemplate; +import org.sosy_lab.common.log.LogManager; + +/** + * TODO use the class to settings or config OR BETTER DELETE IT + * + *

This class is the actual Wrapper around an STP "context" known over there as Validity Checker + * All "context" related operations are handled here Flags, SAT solver settings, STP "environment" + * variables are handled here + * + *

Note the "context" here differs from the 'Context class' + */ +class StpEnvironment { + + private final LogManager logger; + private final ShutdownNotifier shutdownNotifier; + // private final VC vcStpContext; + + public StpEnvironment( + Configuration pConfig, + LogManager pLogger, + ShutdownNotifier pShutdownNotifier, + @Nullable PathCounterTemplate pStpLogfile, + long pRandomSeed) { + + this.logger = pLogger; + this.shutdownNotifier = pShutdownNotifier; + + // create VC (ie context) + // vcStpContext = stpJapi.vc_createValidityChecker(); + + } + + // public long getVC() { + // return StpVC.getEnv(vcStpContext); + // } + +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpFormula.java b/src/org/sosy_lab/java_smt/solvers/stp/StpFormula.java new file mode 100644 index 0000000000..7063e6446f --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpFormula.java @@ -0,0 +1,98 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import com.google.errorprone.annotations.Immutable; +import org.sosy_lab.java_smt.api.ArrayFormula; +import org.sosy_lab.java_smt.api.BitvectorFormula; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.Formula; +import org.sosy_lab.java_smt.api.FormulaType; + +@Immutable +abstract class StpFormula implements Formula { + + private final Expr stpTerm; + + StpFormula(Expr term) { + this.stpTerm = term; + } + + @Override + public final String toString() { + return StpJavaApi.exprString(stpTerm); + } + + @Override + public final boolean equals(Object o) { + if (o == this) { + return true; + } + if (!(o instanceof StpFormula)) { + return false; + } + return stpTerm == ((StpFormula) o).stpTerm; + } + + @Override + public final int hashCode() { + return (int) Expr.getCPtr(stpTerm); + } + + final Expr getTerm() { + return stpTerm; + } + + @Immutable + static final class StpArrayFormula extends StpFormula + implements ArrayFormula { + + private final FormulaType indexType; + private final FormulaType elementType; + + StpArrayFormula(Expr pTerm, FormulaType pIndexType, FormulaType pElementType) { + super(pTerm); + indexType = pIndexType; + elementType = pElementType; + } + + public FormulaType getIndexType() { + return indexType; + } + + public FormulaType getElementType() { + return elementType; + } + } + + @Immutable + static final class StpBitvectorFormula extends StpFormula implements BitvectorFormula { + StpBitvectorFormula(Expr pTerm) { + super(pTerm); + } + } + + @Immutable + static final class StpBooleanFormula extends StpFormula implements BooleanFormula { + StpBooleanFormula(Expr pTerm) { + super(pTerm); + } + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaCreator.java b/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaCreator.java new file mode 100644 index 0000000000..437fd83e87 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaCreator.java @@ -0,0 +1,230 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static com.google.common.base.Preconditions.checkArgument; + +import java.util.Arrays; +import java.util.List; +import org.sosy_lab.java_smt.api.ArrayFormula; +import org.sosy_lab.java_smt.api.BitvectorFormula; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.Formula; +import org.sosy_lab.java_smt.api.FormulaType; +import org.sosy_lab.java_smt.api.FormulaType.FloatingPointType; +import org.sosy_lab.java_smt.api.visitors.FormulaVisitor; +import org.sosy_lab.java_smt.basicimpl.FormulaCreator; +import org.sosy_lab.java_smt.solvers.stp.StpFormula.StpArrayFormula; +import org.sosy_lab.java_smt.solvers.stp.StpFormula.StpBitvectorFormula; +import org.sosy_lab.java_smt.solvers.stp.StpFormula.StpBooleanFormula; + +public class StpFormulaCreator extends FormulaCreator { + + private final VC vc; + + protected StpFormulaCreator(VC vc) { + super(vc, StpJavaApi.vc_boolType(vc), null, null); + this.vc = vc; + } + + @Override + public Type getBitvectorType(int pBitwidth) { + return StpJavaApi.vc_bvType(vc, pBitwidth); + } + + @Override + public Type getFloatingPointType(FloatingPointType pType) { + throw new UnsupportedOperationException("STP does not support FLoating Point yet"); + } + + @Override + public Type getArrayType(Type pIndexType, Type pElementType) { + checkArgument( + StpJavaApi.typeString(pIndexType).contains("BITVECTOR"), "ElementType must be a BITVECTOR"); + checkArgument( + StpJavaApi.typeString(pElementType).contains("BITVECTOR"), + "ElementType must be a BITVECTOR"); + + return StpJavaApi.vc_arrayType(vc, pIndexType, pElementType); + } + + @Override + public Expr makeVariable(Type pType, String pVarName) { + String alphaNum_ = "^[a-zA-Z0-9_]*$"; + checkArgument( + pVarName.matches(alphaNum_), + "A valid Variable Name can only contain Alphanumeric and underscore"); + + return StpJavaApi.vc_varExpr(vc, pVarName, pType); + } + + @Override + public FormulaType getFormulaType(Expr pFormula) { + + switch (StpJavaApi.getType(pFormula)) { + case BOOLEAN_TYPE: + return FormulaType.BooleanType; + case BITVECTOR_TYPE: + int bvTypeSize = StpJavaApi.getBVLength(pFormula); + return FormulaType.getBitvectorTypeWithSize(bvTypeSize); + case ARRAY_TYPE: + + // STP always use BitVector Type + int arrayIndexBitWidth = StpJavaApi.getIWidth(pFormula); + int arrayValueBitWidth = StpJavaApi.getVWidth(pFormula); + + return FormulaType.getArrayType( + FormulaType.getBitvectorTypeWithSize(arrayValueBitWidth), + FormulaType.getBitvectorTypeWithSize(arrayIndexBitWidth)); + + // TODO Resolve this issue https://github.com/stp/stp/issues/333 + // STP always use a BitVector Type to recreate the type get the ValueBitWidth + // and IndexBitWidth of the Expr i.e getIWidth and getVWidth + + case UNKNOWN_TYPE: + throw new IllegalArgumentException("Unknown formula type "); + } + return null; + } + + @Override + public R visit(FormulaVisitor pVisitor, Formula pFormula, Expr pF) { + // TODO I still don't get what this function is trying to do + // TODO implement this + // get the Expr kind for the term + // ... + // return null; + throw new UnsupportedOperationException("Not yet Implemented"); + } + + @Override + public Expr callFunctionImpl(Long pDeclaration, List pArgs) { + // TODO what is this function doing + // return null; + throw new UnsupportedOperationException("Not yet Implemented"); + } + + @Override + public Long declareUFImpl(String pName, Type pReturnType, List pArgTypes) { + + // if pArgTypes is empty then use pReturnType an pName to create a variable + // else ...?! + // TODO Find out about UF + // return null; + throw new UnsupportedOperationException("Not yet Implemented"); + } + + @Override + protected Long getBooleanVarDeclarationImpl(Expr pTFormulaInfo) { + // TODO Auto-generated method stub + return null; + } + + @Override + public Object convertValue(Expr pF) { + // TODO Auto-generated method stub + return null; + } + + /* + * returns true if the Formula is a named variable and not an expression + */ + public boolean isVariable(Expr expr) { + exprkind_t kind = StpJavaApi.getExprKind(expr); + return !kind.equals(exprkind_t.SYMBOL); + } + + @Override + protected Expr extractInfo(Formula pT) { + return StpFormulaManager.getStpTerm(pT); + } + + @Override + protected List extractInfo(List pInput) { + return Arrays.asList(StpFormulaManager.getStpTerm(pInput)); + } + + @SuppressWarnings("unchecked") + @Override + protected FormulaType getFormulaType(T pFormula) { + + Expr term = extractInfo(pFormula); + return (FormulaType) getFormulaType(term); + + // VC vc = getEnv(); + // if (pFormula instanceof BitvectorFormula) { + // long type = msat_term_get_type(extractInfo(pFormula)); + // checkArgument( + // msat_is_bv_type(env, type), + // "BitvectorFormula with actual type " + msat_type_repr(type) + ": " + pFormula); + // return (FormulaType) + // FormulaType.getBitvectorTypeWithSize(msat_get_bv_type_size(env, type)); + // + // } else if (pFormula instanceof FloatingPointFormula) { + // long type = msat_term_get_type(extractInfo(pFormula)); + // checkArgument( + // msat_is_fp_type(env, type), + // "FloatingPointFormula with actual type " + msat_type_repr(type) + ": " + pFormula); + // return (FormulaType) + // FormulaType.getFloatingPointType( + // msat_get_fp_type_exp_width(env, type), msat_get_fp_type_mant_width(env, type)); + // } else if (pFormula instanceof ArrayFormula) { + // FormulaType arrayIndexType = getArrayFormulaIndexType((ArrayFormula) pFormula); + // FormulaType arrayElementType = getArrayFormulaElementType((ArrayFormula) pFormula); + // return (FormulaType) FormulaType.getArrayType(arrayIndexType, arrayElementType); + // } + // return super.getFormulaType(pFormula); + + } + + @Override + public BooleanFormula encapsulateBoolean(Expr pTerm) { + assert getFormulaType(pTerm).isBooleanType(); + return new StpBooleanFormula(pTerm); + } + + @Override + protected BitvectorFormula encapsulateBitvector(Expr pTerm) { + assert getFormulaType(pTerm).isBitvectorType(); + return new StpBitvectorFormula(pTerm); + } + + @Override + protected ArrayFormula encapsulateArray( + Expr pTerm, FormulaType pIndexType, FormulaType pElementType) { + + assert getFormulaType(pTerm).equals(FormulaType.getArrayType(pIndexType, pElementType)); + return new StpArrayFormula<>(pTerm, pIndexType, pElementType); + } + + @Override + protected FormulaType getArrayFormulaIndexType( + ArrayFormula pArray) { + + return ((StpArrayFormula) pArray).getIndexType(); + } + + @Override + protected FormulaType getArrayFormulaElementType( + ArrayFormula pArray) { + + return ((StpArrayFormula) pArray).getElementType(); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaManager.java b/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaManager.java new file mode 100644 index 0000000000..82b8df33b1 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpFormulaManager.java @@ -0,0 +1,70 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import com.google.common.collect.Collections2; +import java.util.Collection; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.sosy_lab.common.Appender; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.Formula; +import org.sosy_lab.java_smt.basicimpl.AbstractFormulaManager; + +public final class StpFormulaManager extends AbstractFormulaManager { + + protected StpFormulaManager( + StpFormulaCreator pFormulaCreator, + StpUFManager pUFManager, + StpBooleanFormulaManager pBooleanManager, + @Nullable StpBitvectorFormulaManager pBitvectorManager, + @Nullable StpArrayFormulaManager pArrayManager) { + super( + pFormulaCreator, + pUFManager, + pBooleanManager, + null, + null, + pBitvectorManager, + null, + null, + pArrayManager); + } + + @Override + public BooleanFormula parse(String pS) throws IllegalArgumentException { + Expr expr = StpJavaApi.vc_parseExpr(getEnvironment(), pS); + return getFormulaCreator().encapsulateBoolean(expr); + } + + static Expr getStpTerm(Formula pT) { + return ((StpFormula) pT).getTerm(); + } + + static Expr[] getStpTerm(Collection pFormulas) { + + return Collections2.transform(pFormulas, StpFormulaManager::getStpTerm).toArray(new Expr[0]); + } + + @Override + public Appender dumpFormula(Expr pT) { + // TODO Auto-generated method stub + return null; + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpModel.java b/src/org/sosy_lab/java_smt/solvers/stp/StpModel.java new file mode 100644 index 0000000000..d15ea83072 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpModel.java @@ -0,0 +1,69 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import com.google.common.base.Preconditions; +import com.google.common.collect.ImmutableList; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.sosy_lab.java_smt.basicimpl.AbstractModel.CachingAbstractModel; + +public class StpModel extends CachingAbstractModel { + + private boolean closed = false; + private final WholeCounterExample model; + private final StpFormulaCreator creator; + private final StpAbstractProver prover; + + protected StpModel( + WholeCounterExample model, StpFormulaCreator pCreator, StpAbstractProver pProver) { + super(pCreator); + this.model = model; + this.creator = pCreator; + this.prover = pProver; + } + + @Override + public void close() { + if (!closed) { + StpJavaApi.vc_deleteWholeCounterExample(model); + closed = true; + } + } + + @Override + protected ImmutableList toList() { + Preconditions.checkState(!closed); + Preconditions.checkState(!prover.closed, "cannot use model after prover is closed"); + ImmutableList.Builder assignments = ImmutableList.builder(); + + int modelSize = StpJavaApi.vc_counterexample_size(creator.getEnv()); + WholeCounterExample[] models = new WholeCounterExample[modelSize]; + + // StpJavaApi.vc_getCounterExampleArray(creator.getEnv(), arg1, arg2, arg3, arg4); + + return null; + } + + @Override + protected @Nullable Expr evalImpl(Expr pFormula) { + // TODO Auto-generated method stub + return null; + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApi.java b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApi.java new file mode 100644 index 0000000000..0bb1ed7eeb --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApi.java @@ -0,0 +1,67 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +// TODO (LATER) use the class to extend StpJavaApi and override funct and give docs +// remove the annoying vc_isBool() from the API, if such functionality is need implement here +// 3. if multiple Type objects are allowed - change "get..." to "create.." otw -> private fields +// 4. +public class StpNativeApi { + + private static final VC vc = StpJavaApi.vc_createValidityChecker(); + + static String getStpVersion() { + return org.sosy_lab.java_smt.solvers.stp.StpJavaApi.get_git_version_tag(); + } + + static VC getStpContextVC() { + return vc; + } + + static long getStpBoolTypePtr() throws Exception { + return Type.getCPtr(StpJavaApi.vc_boolType(vc)); + } + + static Type getStpBoolType(StpSolverContext context) throws Exception { + // VC vc = context.getFormulaCreator().getVC(); + return StpJavaApi.vc_boolType(vc); + } + + static Type getStp32bitsBitVectorType() { + return StpJavaApi.vc_bv32Type(vc); + } + + static Type getStpBitVectorType(int bitSize) { + return StpJavaApi.vc_bvType(vc, bitSize); + } + + static Type getStpArrayType() { + // Long pIndexType, Long pElementType + + // Type x; + // long y = x. + + // get the Types from their addresses - check native for this ? + // confirm they are BV + // then : + // StpJavaApi.vc_arrayType + return StpJavaApi.vc_bv32Type(vc); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest.java b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest.java new file mode 100644 index 0000000000..8928260f54 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest.java @@ -0,0 +1,530 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static org.junit.Assert.assertEquals; + +import org.junit.After; +import org.junit.AssumptionViolatedException; +import org.junit.Before; +import org.junit.BeforeClass; +import org.junit.Ignore; +import org.junit.Test; +import org.sosy_lab.common.NativeLibraries; + +// TODO VERY IMPRTNT : rid this class off sysout; take them ALL out; better branch out Debug +// Sessions +public class StpNativeApiTest { + + private VC stpContextVC; + + @BeforeClass + public static void loadOpensmt2Library() { + try { + NativeLibraries.loadLibrary("stpJapi"); + } catch (UnsatisfiedLinkError e) { + throw new AssumptionViolatedException("Cannot find at the STP native library", e); + } + } + + @Before + public void createStpVC() { + stpContextVC = StpJavaApi.vc_createValidityChecker(); + } + + @After + public void destroyStpVC() { + StpJavaApi.vc_Destroy(stpContextVC); + } + + @Ignore + @Test + public void testStpGitVersion() throws Exception { + String version_sha = StpJavaApi.get_git_version_sha(); + System.out.println("\nSHA of this STP version is :"); + System.out.println(version_sha); + + String version_tag = StpJavaApi.get_git_version_tag(); + System.out.println("\nThis STP version is :"); + System.out.println(version_tag); + } + + @Ignore + @Test + public void testStpCompilationEnvironment() throws Exception { + String compile_env = StpJavaApi.get_compilation_env(); + System.out.println("\nCompilation Environment of this STP version is :"); + + System.out.println(compile_env); + } + + @Ignore // ITS JOB IS DONE; DELETE IT + @Test + public void extendedFunctions() { + // int result = StpJavaApi.extraFunctionSum(100, 50); + Expr expr = StpJavaApi.vc_trueExpr(stpContextVC); + String result = StpJavaApi.getType(expr).name(); + System.out.println("result is " + result); + StpJavaApi.extraSumUpto(10); + System.out.println("Number of Assertion " + StpJavaApi.getNumOfAsserts(stpContextVC) + "\n"); + StpJavaApi.getSomePrinting(stpContextVC); + System.out.println("GOOD MORALS: " + StpJavaApi.getSomeXter(stpContextVC) + "\n"); + } + + @Ignore + @Test + public void testStpSampleFromRepo() throws Exception { + + int width = 8; + + // Register a callback for errors + // StpJavaApi.vc_registerErrorHandler(errorHandler); + + // Create variable "x" + Expr x = StpJavaApi.vc_varExpr(stpContextVC, "x", StpJavaApi.vc_bvType(stpContextVC, width)); + + // Create bitvector x + x + Expr xPlusx = StpJavaApi.vc_bvPlusExpr(stpContextVC, width, x, x); + + // Create bitvector constant 2 + Expr two = StpJavaApi.vc_bvConstExprFromInt(stpContextVC, width, 2); + + // Create bitvector 2*x + Expr xTimes2 = StpJavaApi.vc_bvMultExpr(stpContextVC, width, two, x); + + // Create bool expression x + x = 2*x + Expr equality = StpJavaApi.vc_eqExpr(stpContextVC, xPlusx, xTimes2); + + StpJavaApi.vc_assertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + + // We are asking STP: ∀ x. true → ( x + x = 2*x ) + // This should be VALID. + System.out.println("######First Query\n"); + handleQuery(stpContextVC, equality); + + // We are asking STP: ∀ x. true → ( x + x = 2 ) + // This should be INVALID. + System.out.println("######Second Query\n"); + // Create bool expression x + x = 2 + Expr badEquality = StpJavaApi.vc_eqExpr(stpContextVC, xPlusx, two); + handleQuery(stpContextVC, badEquality); + } + + // void handleQuery(VC handle, Expr queryExpr); + + // Error handler + void errorHandler(String err_msg) throws Exception { + System.out.printf("Error: %s\n", err_msg); + throw new Exception(err_msg); + } + + void handleQuery(VC handle, Expr queryExpr) { + // Print the assertions + System.out.println("Assertions:\n"); + StpJavaApi.vc_printAsserts(handle, 0); + + int result = StpJavaApi.vc_query(handle, queryExpr); + System.out.println("Query:\n"); + StpJavaApi.vc_printQuery(handle); + switch (result) { + case 0: + System.out.println("Query is INVALID\n"); + + // print counter example + System.out.println("Counter example:\n"); + StpJavaApi.vc_printCounterExample(handle); + break; + + case 1: + System.out.println("Query is VALID\n"); + break; + case 2: + System.out.println("Could not answer query\n"); + break; + case 3: + System.out.println("Unhandled error\n"); + } + System.out.println("\n\n"); + } + + @Ignore + @Test + public void createBooleanVariable_() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar", boolType); + Expr boolVar2 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); + // assertNotEquals(-1, StpJavaApi.vc_isBool(boolVar)); + + System.out.println("\n\n ###### ----TYPE INFOR------ ########"); + System.out.println(StpJavaApi.typeString(boolType)); + + // type of the expr BOOLEAN + System.out.println(StpJavaApi.typeString(StpJavaApi.vc_getType(stpContextVC, boolVar))); + // kind of the expr - SYMBOL + System.out.println(StpJavaApi.getExprKind(boolVar)); + + Expr eqExpr = StpJavaApi.vc_eqExpr(stpContextVC, boolVar, boolVar); + System.out.println(StpJavaApi.getExprKind(eqExpr)); + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + Expr eqExpr2 = StpJavaApi.vc_eqExpr(stpContextVC, boolVar, trueExpr); + System.out.println(StpJavaApi.getExprKind(eqExpr2)); + System.out.println("EXPR KIND for trueExpr - " + StpJavaApi.getExprKind(trueExpr)); + + // Always check with getExprKind if expr == SYMBOL + System.out.println(StpJavaApi.exprName(boolVar)); + // System.out.println(StpJavaApi.exprName(eqExpr)); //Raises an exception + + System.out.println("Equqlity Result - " + StpJavaApi.vc_query(stpContextVC, eqExpr)); + + // This will raise exception since eqExpr2 is ill-formed + // because trueExpr is not a valid Term rather a value while boolVar ia a term + // System.out.println("Equality Result - " + StpJavaApi.vc_query(stpContextVC, eqExpr2)); + + Expr expr1 = StpJavaApi.vc_andExpr(stpContextVC, boolVar, trueExpr); + Expr expr2 = StpJavaApi.vc_andExpr(stpContextVC, boolVar, boolVar2); + Expr eqExpr3 = StpJavaApi.vc_eqExpr(stpContextVC, boolVar, expr1); + + Expr andExpr = StpJavaApi.vc_andExpr(stpContextVC, boolVar, eqExpr3); + System.out.println("Evaluation Result expr1 - " + StpJavaApi.vc_query(stpContextVC, expr1)); + System.out.println("Evaluation Result eqExpr3 - " + StpJavaApi.vc_query(stpContextVC, eqExpr3)); + System.out.println("Evaluation Result andExpr - " + StpJavaApi.vc_query(stpContextVC, andExpr)); + System.out.println("EXPR KIND for expr1 - " + StpJavaApi.getExprKind(expr1)); + System.out.println("EXPR KIND for expr2 - " + StpJavaApi.getExprKind(expr2)); + // System.out.println("IF expr2 is a SYMBOL, it has a name: " + StpJavaApi.exprName(expr2)); + // //Error - non symbol + // assertEquals(boolType, StpJavaApi.vc_getType(vc, boolVar)); + } + + @Test + public void createBooleanVariable() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar", boolType); + + assertEquals("BOOLEAN", StpJavaApi.typeString(boolType).trim()); + assertEquals(exprkind_t.SYMBOL, StpJavaApi.getExprKind(boolVar)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, boolVar)); + } + + @Test + public void validateNotOnBooleanValues() { + Expr checkforFalseValue = + StpJavaApi.vc_notExpr(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + Expr checkforTrueValue = + StpJavaApi.vc_notExpr(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, checkforFalseValue)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, checkforTrueValue)); + } + + @Test + public void validateAndOnBooleanValues() { + Expr trueANDfalse = + StpJavaApi.vc_andExpr( + stpContextVC, + StpJavaApi.vc_trueExpr(stpContextVC), + StpJavaApi.vc_falseExpr(stpContextVC)); + Expr trueANDtrue = + StpJavaApi.vc_andExpr( + stpContextVC, + StpJavaApi.vc_trueExpr(stpContextVC), + StpJavaApi.vc_trueExpr(stpContextVC)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, trueANDfalse)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, trueANDtrue)); + } + + @Test + public void validateOrOnBooleanValues() { + Expr falseORfalse = + StpJavaApi.vc_orExpr( + stpContextVC, + StpJavaApi.vc_falseExpr(stpContextVC), + StpJavaApi.vc_falseExpr(stpContextVC)); + Expr trueORfalse = + StpJavaApi.vc_orExpr( + stpContextVC, + StpJavaApi.vc_trueExpr(stpContextVC), + StpJavaApi.vc_falseExpr(stpContextVC)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, falseORfalse)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, trueORfalse)); + } + + @Ignore + @Test(expected = RuntimeException.class) + public void validateEqualityOnBooleanValues() { + Expr falseORfalse = + StpJavaApi.vc_eqExpr( + stpContextVC, + StpJavaApi.vc_falseExpr(stpContextVC), + StpJavaApi.vc_falseExpr(stpContextVC)); + Expr trueORfalse = + StpJavaApi.vc_eqExpr( + stpContextVC, + StpJavaApi.vc_trueExpr(stpContextVC), + StpJavaApi.vc_falseExpr(stpContextVC)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, falseORfalse)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, trueORfalse)); + + // assertThrows(RuntimeException.class,()->{ StpJavaApi.vc_query_with_timeout(stpContextVC, + // trueORfalse, 1, 360);} ); + // assertEquals(1, StpJavaApi.vc_query_with_timeout(stpContextVC, trueORfalse, 1, 3)); + + } + + @Ignore + @Test + public void validateEqualityOnBooleanFormula1() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar", boolType); + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar); + + Expr falseEQtrue = StpJavaApi.vc_eqExpr(stpContextVC, boolVar, notboolVar); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, falseEQtrue)); + } + + @Test + public void validateEqualityOnBooleanFormula2() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar", boolType); // false + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar); + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + + Expr expr1 = StpJavaApi.vc_andExpr(stpContextVC, boolVar, trueExpr); // false + Expr expr2 = StpJavaApi.vc_andExpr(stpContextVC, notboolVar, notboolVar); // true + + Expr boolVarEQexpr1 = StpJavaApi.vc_eqExpr(stpContextVC, boolVar, expr1); + Expr notboolVarEQexpr2 = StpJavaApi.vc_eqExpr(stpContextVC, notboolVar, expr2); + + assertEquals(1, StpJavaApi.vc_query(stpContextVC, boolVarEQexpr1)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, notboolVarEQexpr2)); + } + + @Ignore // TODO DEBUG THIS + @Test(expected = RuntimeException.class) + public void validateEqualityOnBooleanFormulaGivesException() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar); + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + + Expr expr1 = StpJavaApi.vc_andExpr(stpContextVC, boolVar, trueExpr); // false + Expr expr2 = StpJavaApi.vc_andExpr(stpContextVC, notboolVar, notboolVar); // <- bug here + + // The two Expression Formulae must have the same type. + assertEquals(type_t.BOOLEAN_TYPE, StpJavaApi.getType(expr1)); + assertEquals(type_t.BOOLEAN_TYPE, StpJavaApi.getType(expr2)); + + assertEquals( + "BOOLEAN", StpJavaApi.typeString(StpJavaApi.vc_getType(stpContextVC, expr1)).trim()); + assertEquals( + "BOOLEAN", StpJavaApi.typeString(StpJavaApi.vc_getType(stpContextVC, expr2)).trim()); + + Expr expr1EQexpr2 = StpJavaApi.vc_eqExpr(stpContextVC, expr1, expr2); + + assertEquals(1, StpJavaApi.vc_query(stpContextVC, expr1EQexpr2)); // <-exception raised here + + // STP itself exits (i guess). So no way round it + /* + * Fatal Error: TransformTerm: Illegal kind: You have input a nonterm: 499:(NOT 498:boolVar2) 39 + */ + } + + @Test + public void validateXor() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); // false + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar); // true + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + Expr falseExpr = StpJavaApi.vc_falseExpr(stpContextVC); + + Expr shouldBeTrue1 = StpJavaApi.vc_xorExpr(stpContextVC, boolVar, notboolVar); + Expr shouldBeTrue2 = StpJavaApi.vc_xorExpr(stpContextVC, trueExpr, falseExpr); + Expr shouldBeFalse1 = StpJavaApi.vc_xorExpr(stpContextVC, falseExpr, falseExpr); + Expr shouldBeFalse2 = StpJavaApi.vc_xorExpr(stpContextVC, notboolVar, notboolVar); + + assertEquals(1, StpJavaApi.vc_query(stpContextVC, shouldBeTrue1)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, shouldBeTrue2)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, shouldBeFalse1)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, shouldBeFalse2)); + } + + @Test + public void validateIfonlyIf() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); // false + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar); // true + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + Expr falseExpr = StpJavaApi.vc_falseExpr(stpContextVC); + + Expr shouldBeFalse1 = StpJavaApi.vc_iffExpr(stpContextVC, boolVar, notboolVar); + Expr shouldBeFalse2 = StpJavaApi.vc_iffExpr(stpContextVC, trueExpr, falseExpr); + Expr shouldBeTrue1 = StpJavaApi.vc_iffExpr(stpContextVC, falseExpr, falseExpr); + Expr shouldBeTrue2 = StpJavaApi.vc_iffExpr(stpContextVC, notboolVar, notboolVar); + + assertEquals(1, StpJavaApi.vc_query(stpContextVC, shouldBeTrue1)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, shouldBeTrue2)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, shouldBeFalse1)); + assertEquals(0, StpJavaApi.vc_query(stpContextVC, shouldBeFalse2)); + } + + @Test + public void validateIfThenElse_BoolOnly() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar1 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar1", boolType); // false + Expr boolVar2 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); // false + // Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar1);// true + + Expr trueExpr = StpJavaApi.vc_trueExpr(stpContextVC); + Expr falseExpr = StpJavaApi.vc_falseExpr(stpContextVC); + + Expr shouldBeboolVar1 = StpJavaApi.vc_iteExpr(stpContextVC, trueExpr, boolVar1, boolVar2); + Expr shouldBeboolVar2 = StpJavaApi.vc_iteExpr(stpContextVC, falseExpr, boolVar1, boolVar2); + + assertEquals("boolVar1", StpJavaApi.exprName(shouldBeboolVar1)); + assertEquals("boolVar2", StpJavaApi.exprName(shouldBeboolVar2)); + + // TODO: extend test for: + // validateIfThenElse_BVOnly + // validateIfThenElse_BoolAndBV + } + + @Ignore // because it stops the tests + @Test(expected = RuntimeException.class) // This exception cannot be caught the STP lib exits + public void validateIfThenElse_ConditionNotBool_Exception() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Type bv32Type = StpJavaApi.vc_bv32Type(stpContextVC); + + Expr boolVar1 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar1", boolType); // false + Expr boolVar2 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar2", boolType); // false + Expr bv32Var = StpJavaApi.vc_varExpr(stpContextVC, "bv32Var", bv32Type); + + Expr shouldBeboolVar1 = StpJavaApi.vc_iteExpr(stpContextVC, bv32Var, boolVar1, boolVar2); + + assertEquals("boolVar1", StpJavaApi.exprName(shouldBeboolVar1)); + } + + @Test + public void createBitVectVariables() { + int numOfBits = 8; + Type bvType = StpJavaApi.vc_bvType(stpContextVC, numOfBits); + Type bv32Type = StpJavaApi.vc_bv32Type(stpContextVC); + + Expr bv8bitVar = StpJavaApi.vc_varExpr(stpContextVC, "bv8bitVar", bvType); + Expr bv32bitVar = StpJavaApi.vc_varExpr(stpContextVC, "bv32bitVar", bv32Type); + + assertEquals(exprkind_t.SYMBOL, StpJavaApi.getExprKind(bv8bitVar)); + assertEquals(exprkind_t.SYMBOL, StpJavaApi.getExprKind(bv32bitVar)); + + assertEquals("bv8bitVar", StpJavaApi.exprName(bv8bitVar)); + assertEquals("bv32bitVar", StpJavaApi.exprName(bv32bitVar)); + + assertEquals( + "BITVECTOR(00000008)", + StpJavaApi.typeString(StpJavaApi.vc_getType(stpContextVC, bv8bitVar)).trim()); + + assertEquals( + "BITVECTOR(00000020)", + StpJavaApi.typeString(StpJavaApi.vc_getType(stpContextVC, bv32bitVar)).trim()); + + assertEquals(8, StpJavaApi.vc_getBVLength(stpContextVC, bv8bitVar)); + assertEquals(32, StpJavaApi.vc_getBVLength(stpContextVC, bv32bitVar)); + } + + @Test + public void convertBoolToBitVector_() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar1 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar1", boolType); // false + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar1); // true + + Expr bvFrom_boolVar1 = StpJavaApi.vc_boolToBVExpr(stpContextVC, boolVar1); // 1 + Expr bvFrom_notboolVar1 = StpJavaApi.vc_boolToBVExpr(stpContextVC, notboolVar); + + Type type1 = StpJavaApi.vc_getType(stpContextVC, bvFrom_boolVar1); + type_t type2 = StpJavaApi.getType(bvFrom_notboolVar1); + + // System.out.println("TYPE-1 - " + StpJavaApi.typeString(type1)); + // System.out.println("TYPE-2 - " + type2.name()); + } + + @Ignore + @Test + public void convertBoolToBitVector() { + + Type boolType = StpJavaApi.vc_boolType(stpContextVC); + Expr boolVar1 = StpJavaApi.vc_varExpr(stpContextVC, "boolVar1", boolType); // false + Expr notboolVar = StpJavaApi.vc_notExpr(stpContextVC, boolVar1); // true + + Expr bvFrom_boolVar1 = StpJavaApi.vc_boolToBVExpr(stpContextVC, boolVar1); + Expr bvFrom_notboolVar1 = StpJavaApi.vc_boolToBVExpr(stpContextVC, notboolVar); + + Expr oneBVconst = StpJavaApi.vc_bv32ConstExprFromInt(stpContextVC, 1); + Expr zeroBVconst = StpJavaApi.vc_bv32ConstExprFromInt(stpContextVC, 0); + // Expr zeroBVconst = StpJavaApi.vc_bvConstExprFromStr(stpContextVC, "0"); + + // int width = 32; + // Expr x = StpJavaApi.vc_varExpr(stpContextVC, "x", StpJavaApi.vc_bvType(stpContextVC, width)); + // // Create bitvector one*x + // Expr xTimesOne = StpJavaApi.vc_bvMultExpr(stpContextVC, width, oneBVconst, x); + // Expr xTimesBv1 = StpJavaApi.vc_bvMultExpr(stpContextVC, width, bvFrom_boolVar1, x); + // + // // Create bitvector zero*x + // Expr xTimesZero = StpJavaApi.vc_bvMultExpr(stpContextVC, width, zeroBVconst, x); + // Expr xTimesBv2 = StpJavaApi.vc_bvMultExpr(stpContextVC, width, bvFrom_notboolVar1, x); + + Expr equality1 = StpJavaApi.vc_eqExpr(stpContextVC, oneBVconst, bvFrom_boolVar1); + Expr equality2 = StpJavaApi.vc_eqExpr(stpContextVC, zeroBVconst, bvFrom_notboolVar1); + + assertEquals(1, StpJavaApi.vc_query(stpContextVC, equality1)); + assertEquals(1, StpJavaApi.vc_query(stpContextVC, equality2)); + + // TODO : what does these do: + // StpJavaApi.vc_assertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + + // Print the assertions + // System.out.println("Assertions:\n"); + // StpJavaApi.vc_printAsserts(handle, 0); + + } + + // TODO Extend test: vc_arrayType, vc_paramBoolExpr + + @Test + public void createArrayVariable() { + + // vc_arrayType(VC vc, Type typeIndex, Type typeData); + // index type and data type must both be of bitvector (bv) type. + } + + // vc_isBool -BAD + +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest2.java b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest2.java new file mode 100644 index 0000000000..33a6cde24a --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpNativeApiTest2.java @@ -0,0 +1,383 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static org.junit.Assert.assertEquals; + +import org.junit.After; +import org.junit.AssumptionViolatedException; +import org.junit.Before; +import org.junit.BeforeClass; +import org.junit.Ignore; +import org.junit.Test; +import org.sosy_lab.common.NativeLibraries; + +public class StpNativeApiTest2 { + + private VC stpContextVC; + + @BeforeClass + public static void loadOpensmt2Library() { + try { + NativeLibraries.loadLibrary("stpJapi"); + } catch (UnsatisfiedLinkError e) { + throw new AssumptionViolatedException("Cannot find at the STP native library", e); + } + } + + @Before + public void createStpVC() { + stpContextVC = StpJavaApi.vc_createValidityChecker(); + } + + @After + public void destroyStpVC() { + StpJavaApi.vc_Destroy(stpContextVC); + } + + // BITVECTORS + int width = 8; + Expr bv_x, bv_xPlusx, two, twenty, xTimes2, xTimes20; + Expr bv_equality, bv_equality2, bv_non_equality; + + // BOOLEAN + Expr x1, x2, x3; + Expr notx1, notx2, notx3; + Expr x1ORnotx2, notx1ORx2; + + ////// VARIABLE CREATION TESTS ///////////////////// + + @Test + public void createBVvariables() { + // Create variable "x" + bv_x = StpJavaApi.vc_varExpr(stpContextVC, "x", StpJavaApi.vc_bvType(stpContextVC, width)); + + // Create bitvector constant 2 + two = StpJavaApi.vc_bvConstExprFromInt(stpContextVC, width, 2); + twenty = StpJavaApi.vc_bvConstExprFromInt(stpContextVC, width, 20); + } + + @Test + public void createBOOLvariables() { + x1 = StpJavaApi.vc_varExpr(stpContextVC, "x1", StpJavaApi.vc_boolType(stpContextVC)); + x2 = StpJavaApi.vc_varExpr(stpContextVC, "x2", StpJavaApi.vc_boolType(stpContextVC)); + x3 = StpJavaApi.vc_varExpr(stpContextVC, "x3", StpJavaApi.vc_boolType(stpContextVC)); + } + + @Test + public void createARRAYvariables() { + // x1 = StpJavaApi.vc_varExpr(stpContextVC, "x1", StpJavaApi.vc_boolType(stpContextVC)); + // x2 = StpJavaApi.vc_varExpr(stpContextVC, "x2", StpJavaApi.vc_boolType(stpContextVC)); + // x3 = StpJavaApi.vc_varExpr(stpContextVC, "x3", StpJavaApi.vc_boolType(stpContextVC)); + } + + @Test + public void getTypesofVariables() { + createBVvariables(); + createBOOLvariables(); + + assertEquals(type_t.BOOLEAN_TYPE, StpJavaApi.getType(x1)); + assertEquals(type_t.BOOLEAN_TYPE, StpJavaApi.getType(x2)); + assertEquals(type_t.BOOLEAN_TYPE, StpJavaApi.getType(x3)); + assertEquals(type_t.BITVECTOR_TYPE, StpJavaApi.getType(bv_x)); + assertEquals(type_t.BITVECTOR_TYPE, StpJavaApi.getType(two)); + assertEquals(type_t.BITVECTOR_TYPE, StpJavaApi.getType(twenty)); + } + + @Test + public void createBVformulars() { + createBVvariables(); + + // Create bitvector x + x + bv_xPlusx = StpJavaApi.vc_bvPlusExpr(stpContextVC, width, bv_x, bv_x); // Non-Formla can't + // assert + + // Create bitvector 2*x + xTimes2 = StpJavaApi.vc_bvMultExpr(stpContextVC, width, two, bv_x); // Non-Formla can't assert + xTimes20 = StpJavaApi.vc_bvMultExpr(stpContextVC, width, twenty, bv_x); // Non-Formla can't + // assert + + // Create bool expression x + x = 2*x + bv_equality = StpJavaApi.vc_eqExpr(stpContextVC, bv_xPlusx, xTimes2); + bv_equality2 = StpJavaApi.vc_eqExpr(stpContextVC, bv_xPlusx, xTimes20); + + bv_non_equality = StpJavaApi.vc_notExpr(stpContextVC, bv_equality2); + } + + @Test + public void createBOOLformulars() { + createBOOLvariables(); + + notx1 = StpJavaApi.vc_notExpr(stpContextVC, x1); + notx2 = StpJavaApi.vc_notExpr(stpContextVC, x2); + notx3 = StpJavaApi.vc_notExpr(stpContextVC, x3); + + x1ORnotx2 = StpJavaApi.vc_orExpr(stpContextVC, x1, notx2); + notx1ORx2 = StpJavaApi.vc_orExpr(stpContextVC, notx1, x2); + } + + private String evaluateVCqueryResult(int result) { + + if (result == 0) { + return "INVALID"; + } else if (result == 1) { + return "VALID"; + } else if (result == 2) { + return "AN ERROR HAS OCCURED"; + } else if (result == 3) { + return "TIMEOUT REACHED"; + } else { + return "UNKNOWN CODE - FATAL ERROR !!!"; + } + } + + @Test + public void validBOOLexpr() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + Expr frmlr1 = StpJavaApi.vc_andExpr(stpContextVC, term1, notx1ORx2); + + // int result = StpJavaApi.vc_query(stpContextVC, frmlr1); //MYSTERY: why does this gives + // invalid + + StpJavaApi.addAssertFormula(stpContextVC, frmlr1); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + System.out.println("RESULT IS : " + evaluateVCqueryResult(result)); + + // StpJavaApi.vc_printAsserts(stpContextVC, 0); + + } + + StpEnv env; + + void setupStpEnv(VC vc) { + env = StpJavaApi.createStpEnv(vc); + } + + @Test + public void validBOOLeprWithAssert() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.addAssertFormula(stpContextVC, notx1ORx2); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + System.out.println("ASSERT RESULT IS : " + evaluateVCqueryResult(result)); + } + + @Ignore + @Test + public void invalidBOOLexpr1() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.addAssertFormula(stpContextVC, notx1); + StpJavaApi.addAssertFormula(stpContextVC, x2); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + System.out.println("INVALID ASSERT RESULT - 1 IS : " + evaluateVCqueryResult(result)); // VALID + } + + @Ignore + @Test + public void invalidBOOLexpr2() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.addAssertFormula(stpContextVC, notx1); + StpJavaApi.addAssertFormula(stpContextVC, x2); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("INVALID ASSERT RESULT - 2 IS : " + evaluateVCqueryResult(result)); + + // print counter example + System.out.println("Counter example:\n"); + StpJavaApi.vc_printCounterExample(stpContextVC); + } + + @Ignore + @Test + public void ext_BOOLexpr2() { + createBOOLvariables(); + createBOOLformulars(); + setupStpEnv(stpContextVC); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + + StpJavaApi.ext_push(env); + StpJavaApi.ext_addFormula(env, term1); + StpJavaApi.ext_push(env); + StpJavaApi.ext_addFormula(env, notx1); + StpJavaApi.ext_push(env); + StpJavaApi.ext_addFormula(env, x2); + + System.out.println("CACHE size is " + StpJavaApi.getCacheSize(env)); + StpJavaApi.ext_checkSat(env); + } + + @Ignore + @Test + public void validBOOLexpr3() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, notx1); + // StpJavaApi.push(stpContextVC); + // StpJavaApi.addAssertFormula(stpContextVC, x2); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("3) INVALID ASSERT RESULT - IS : " + evaluateVCqueryResult(result)); + } + + @Ignore + @Test + public void pushAndPop_invalidBOOLexpr2() { + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, notx1); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, x2); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("2) INVALID ASSERT RESULT - IS : " + evaluateVCqueryResult(result)); + + // print counter example + // System.out.println("Counter example:\n"); + // StpJavaApi.vc_printCounterExample(stpContextVC); + + // System.out.println("\n" + "ASSERTS:"); + // StpJavaApi.vc_printAsserts(stpContextVC, 0); + + StpJavaApi.pop(stpContextVC); + System.out.println("\n" + "AFTER POP 1"); + int result2 = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("2) INVALID ASSERT RESULT - IS : " + evaluateVCqueryResult(result2)); + + System.out.println("\n" + "ASSERTS:"); + StpJavaApi.vc_printAsserts(stpContextVC, 0); + + StpJavaApi.pop(stpContextVC); + System.out.println("\n" + "AFTER POP 2"); + int result3 = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("2) INVALID ASSERT RESULT - IS : " + evaluateVCqueryResult(result3)); + + // System.out.println("\n" + "ASSERTS:"); + // StpJavaApi.vc_printAsserts(stpContextVC, 0); + + System.out.println("\n" + "END OF POP TESTING" + "\n"); + } + + @Ignore + @Test + public void addAssertionsBV() { + + createBVvariables(); + createBVformulars(); + + // StpJavaApi.push(stpContextVC); // Why does this ASSERT (TRUE) + // StpJavaApi.addAssertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + // StpJavaApi.addAssertFormula(stpContextVC, equality); + + // StpJavaApi.ext_AssertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + // StpJavaApi.ext_AssertFormula(stpContextVC, equality); + + StpJavaApi.addAssertFormula(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + + // StpJavaApi.vc_printAsserts(stpContextVC, 0); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("RESULT IS : " + result); + + StpJavaApi.vc_printAsserts(stpContextVC, 0); + + // StpJavaApi.checkSAT_old(stpContextVC); + // StpJavaApi.checkSAT(stpContextVC, notEq); + } + + @Ignore + @Test + public void testingSAT() { + createBVvariables(); + createBVformulars(); + + StpJavaApi.vc_assertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + StpJavaApi.vc_assertFormula(stpContextVC, bv_equality); + + // StpJavaApi.ext_AssertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + // StpJavaApi.ext_AssertFormula(stpContextVC, equality); + + StpJavaApi.extraSumUpto(19); + + int result = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + + // Print the assertions + System.out.println("Assertions: ===> "); + StpJavaApi.vc_printAsserts(stpContextVC, 0); + // System.out.println("\nQuery: ===> "); + // StpJavaApi.vc_printQuery(stpContextVC); + + System.out.println("VC_QUERY RESULT : ===> " + result); + + StpJavaApi.ext_AssertFormula(stpContextVC, StpJavaApi.vc_trueExpr(stpContextVC)); + } + + @Test + public void testingUNSAT() { + StpJavaApi.push(stpContextVC); + + // create & add Formula + createBOOLvariables(); + createBOOLformulars(); + + Expr term1 = StpJavaApi.vc_orExpr(stpContextVC, x1ORnotx2, x3); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, term1); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, notx1); + StpJavaApi.push(stpContextVC); + StpJavaApi.addAssertFormula(stpContextVC, x2); + + int result3 = StpJavaApi.vc_query(stpContextVC, StpJavaApi.vc_falseExpr(stpContextVC)); + System.out.println("2) INVALID ASSERT RESULT - IS : " + evaluateVCqueryResult(result3)); + + String result = StpJavaApi.getAllModel(stpContextVC); + System.out.println(result); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpSolverContext.java b/src/org/sosy_lab/java_smt/solvers/stp/StpSolverContext.java new file mode 100644 index 0000000000..b6e5e886f6 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpSolverContext.java @@ -0,0 +1,157 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import java.util.Set; +import java.util.logging.Level; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.junit.AssumptionViolatedException; +import org.sosy_lab.common.NativeLibraries; +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.common.configuration.Configuration; +import org.sosy_lab.common.io.PathCounterTemplate; +import org.sosy_lab.common.log.LogManager; +import org.sosy_lab.java_smt.SolverContextFactory.Solvers; +import org.sosy_lab.java_smt.api.InterpolatingProverEnvironment; +import org.sosy_lab.java_smt.api.OptimizationProverEnvironment; +import org.sosy_lab.java_smt.api.ProverEnvironment; +import org.sosy_lab.java_smt.basicimpl.AbstractSolverContext; + +public final class StpSolverContext extends AbstractSolverContext { + + // private StpFormulaManager manager; + private final StpFormulaCreator formulaCreator; + private final LogManager logger; + private final ShutdownNotifier shutdownNotifier; + + // Context is Validity Checker (VC) in STP + // private final StpVC stpContext; + private static final VC vcStpContext; + + static { + // load the shared library + try { + NativeLibraries.loadLibrary("stpJapi"); + vcStpContext = StpNativeApi.getStpContextVC(); + } catch (UnsatisfiedLinkError e) { + throw new AssumptionViolatedException("Cannot find at the STP native library", e); + } + } + + private StpSolverContext( + StpFormulaManager pFormulaMgr, + StpFormulaCreator pFormulaCreator, + LogManager pLogger, + ShutdownNotifier pShutdownNotifier) { + super(pFormulaMgr); + + this.formulaCreator = pFormulaCreator; + this.logger = pLogger; + this.shutdownNotifier = pShutdownNotifier; + } + + public static StpSolverContext create( + Configuration config, + LogManager logger, + ShutdownNotifier shutdownNotifier, + @Nullable PathCounterTemplate stpLogfile, + long randomSeed) { + // + // //load the shared library + // try { + // NativeLibraries.loadLibrary("stpJapi"); + // } catch (UnsatisfiedLinkError e) { + // throw new AssumptionViolatedException("Cannot find at the STP native library", e); + // } + + // Create or setup the 'environment' with supplied parameters and other java-smt defaults + // vcStpContext // this is the 'env' + + // use the 'environment' to create a FormulaCreator object + StpFormulaCreator formulaCreator = new StpFormulaCreator(vcStpContext); + + // use the FormulaCreator object to create FormulaManager object for all supported Theories + StpBooleanFormulaManager booleanFrmMgr = new StpBooleanFormulaManager(formulaCreator); + StpUFManager functionFrmMgr = new StpUFManager(formulaCreator); + StpArrayFormulaManager arrayFrmMgr = new StpArrayFormulaManager(formulaCreator); + StpBitvectorFormulaManager bitvectorFrmMgr = new StpBitvectorFormulaManager(formulaCreator); + + // Create the main FormulaManager to manage all supported Formula types + StpFormulaManager formulaMgr = + new StpFormulaManager( + formulaCreator, functionFrmMgr, booleanFrmMgr, bitvectorFrmMgr, arrayFrmMgr); + + // Create the SolverContext with the FormulaCreator and main FormulaManager Objects + return new StpSolverContext(formulaMgr, formulaCreator, logger, shutdownNotifier); + } + + public StpFormulaCreator getFormulaCreator() { + return this.formulaCreator; + } + + @Override + public String getVersion() { + return StpNativeApi.getStpVersion(); + } + + @Override + public Solvers getSolverName() { + return Solvers.STP; + } + + @Override + public void close() { + logger.log(Level.FINER, "Freeing STP environment resources"); + // stpJapi.vc_Destroy(formulaCreator.getEnv()); //TODO: use this and make vcStpContext + // non-static + StpJavaApi.vc_Destroy(vcStpContext); + } + + @Override + protected ProverEnvironment newProverEnvironment0(Set proverOptions) { + return new StpTheoremProver(this, shutdownNotifier, formulaCreator, proverOptions); + } + + @Override + protected InterpolatingProverEnvironment newProverEnvironmentWithInterpolation0( + Set pSet) { + // TODO stub + throw new UnsupportedOperationException("Interpolating Prover for STP is not implemented yet"); + } + + @Override + protected OptimizationProverEnvironment newOptimizationProverEnvironment0( + Set proverOptions) { + + // TODO I need to confirm about this + throw new UnsupportedOperationException("Support for optimization in STP not implemented yet"); + } + + @Override + protected boolean supportsAssumptionSolving() { + // TODO I am not sure about this + return false; + } + + public VC createEnvironment(Long conf) { + // create a new VC context or reuse existing one + return formulaCreator.getEnv(); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpTheoremProver.java b/src/org/sosy_lab/java_smt/solvers/stp/StpTheoremProver.java new file mode 100644 index 0000000000..a045882e30 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpTheoremProver.java @@ -0,0 +1,53 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import com.google.common.base.Preconditions; +import java.util.Set; +import org.checkerframework.checker.nullness.qual.Nullable; +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.ProverEnvironment; +import org.sosy_lab.java_smt.api.SolverContext.ProverOptions; + +class StpTheoremProver extends StpAbstractProver implements ProverEnvironment { + + protected StpTheoremProver( + StpSolverContext pContext, + ShutdownNotifier pShutdownNotifier, + StpFormulaCreator pFrmcreator, + Set pOptions) { + super(pContext, pOptions, pFrmcreator, pShutdownNotifier); + } + + @Override + public @Nullable Void addConstraint(BooleanFormula pConstraint) throws InterruptedException { + Preconditions.checkState(!closed); + + StpJavaApi.addAssertFormula(currVC, StpFormulaManager.getStpTerm(pConstraint)); + return null; + } + + @Override + public void push() { + Preconditions.checkState(!closed); + StpJavaApi.vc_push(currVC); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/StpUFManager.java b/src/org/sosy_lab/java_smt/solvers/stp/StpUFManager.java new file mode 100644 index 0000000000..9e27279913 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/StpUFManager.java @@ -0,0 +1,29 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import org.sosy_lab.java_smt.basicimpl.AbstractUFManager; + +class StpUFManager extends AbstractUFManager { + + protected StpUFManager(StpFormulaCreator pCreator) { + super(pCreator); + } +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/TestStpSolver.java b/src/org/sosy_lab/java_smt/solvers/stp/TestStpSolver.java new file mode 100644 index 0000000000..c4a09f4f11 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/TestStpSolver.java @@ -0,0 +1,293 @@ +/* + * JavaSMT is an API wrapper for a collection of SMT solvers. + * This file is part of JavaSMT. + * + * Copyright (C) 2007-2019 Dirk Beyer + * All rights reserved. + * + * Licensed under the Apache License, Version 2.0 (the "License"); + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ +package org.sosy_lab.java_smt.solvers.stp; + +import static org.junit.Assert.assertEquals; +import static org.junit.Assert.assertFalse; +import static org.junit.Assert.assertTrue; + +import org.junit.AfterClass; +import org.junit.Ignore; +import org.junit.Test; +import org.sosy_lab.common.ShutdownNotifier; +import org.sosy_lab.common.configuration.Configuration; +import org.sosy_lab.common.configuration.InvalidConfigurationException; +import org.sosy_lab.common.log.BasicLogManager; +import org.sosy_lab.common.log.LogManager; +import org.sosy_lab.java_smt.SolverContextFactory; +import org.sosy_lab.java_smt.SolverContextFactory.Solvers; +import org.sosy_lab.java_smt.api.ArrayFormula; +import org.sosy_lab.java_smt.api.ArrayFormulaManager; +import org.sosy_lab.java_smt.api.BitvectorFormula; +import org.sosy_lab.java_smt.api.BitvectorFormulaManager; +import org.sosy_lab.java_smt.api.BooleanFormula; +import org.sosy_lab.java_smt.api.BooleanFormulaManager; +import org.sosy_lab.java_smt.api.FormulaType; +import org.sosy_lab.java_smt.api.FormulaType.BitvectorType; +import org.sosy_lab.java_smt.api.ProverEnvironment; +import org.sosy_lab.java_smt.api.SolverContext; +import org.sosy_lab.java_smt.api.SolverContext.ProverOptions; +import org.sosy_lab.java_smt.api.SolverException; + +public class TestStpSolver { + + private final Configuration config; + private final LogManager logger; + private final ShutdownNotifier shutdownNotifier; + private final Solvers solver; + + public static SolverContext context; + + // variables + BooleanFormula p; + BooleanFormula q; + BooleanFormula trueValue; + BooleanFormula falseValue; + + BitvectorFormula bv8; + BitvectorFormula bv32; + BitvectorFormula zero; + BitvectorFormula two; + BitvectorFormula twenty; + BitvectorFormula x; + BitvectorFormula y; + + ArrayFormula arrayOfBV8; + ArrayFormula arrayOfBV32; + + public TestStpSolver() throws InvalidConfigurationException { + config = Configuration.defaultConfiguration(); + logger = BasicLogManager.create(config); + shutdownNotifier = ShutdownNotifier.createDummy(); + solver = Solvers.STP; + // solver = Solvers.MATHSAT5; + + context = SolverContextFactory.createSolverContext(config, logger, shutdownNotifier, solver); + } + + @AfterClass + public static void afterClass() { + if (context != null) { + context.close(); + } + } + + @Test + public void createBooleanVariables() { + + BooleanFormulaManager boolFmgr = context.getFormulaManager().getBooleanFormulaManager(); + + p = boolFmgr.makeVariable("p"); + q = boolFmgr.makeVariable("q"); + trueValue = boolFmgr.makeTrue(); + falseValue = boolFmgr.makeBoolean(false); + + assertTrue(context.getFormulaManager().getFormulaType(p).isBooleanType()); + assertTrue(context.getFormulaManager().getFormulaType(q).isBooleanType()); + assertTrue(context.getFormulaManager().getFormulaType(trueValue).isBooleanType()); + assertTrue(context.getFormulaManager().getFormulaType(falseValue).isBooleanType()); + + // these would raise a nasty exception because these are not Values + // assertTrue(boolFMgr.isFalse(falseVar)); + // assertTrue(boolFMgr.isTrue(trueVar)); + + assertTrue(boolFmgr.isTrue(trueValue)); + assertTrue(boolFmgr.isFalse(falseValue)); + } + + @Test + public void createBitVectorVariables() { + + BitvectorFormulaManager bvFmgr = context.getFormulaManager().getBitvectorFormulaManager(); + + bv8 = bvFmgr.makeVariable(8, "bv8"); + bv32 = bvFmgr.makeVariable(32, "bv32"); + x = bvFmgr.makeVariable(8, "x"); + y = bvFmgr.makeVariable(8, "y"); + + // these are constants + zero = bvFmgr.makeBitvector(8, 0); + two = bvFmgr.makeBitvector(8, 2); + twenty = bvFmgr.makeBitvector(8, 20); + + assertTrue(context.getFormulaManager().getFormulaType(bv8).isBitvectorType()); + assertTrue(context.getFormulaManager().getFormulaType(bv32).isBitvectorType()); + } + + @Test + public void createArrayVariables() { + + ArrayFormulaManager arrayFmgr = context.getFormulaManager().getArrayFormulaManager(); + + BitvectorType indexType8 = FormulaType.getBitvectorTypeWithSize(8); + BitvectorType indexType32 = FormulaType.getBitvectorTypeWithSize(32); + BitvectorType elementType8 = FormulaType.getBitvectorTypeWithSize(8); + BitvectorType elementType32 = FormulaType.getBitvectorTypeWithSize(32); + + // Same bit size is expected by javaSMT otherwise you get "expected Array by got Array" + arrayOfBV8 = arrayFmgr.makeArray("array_8", indexType8, elementType8); + arrayOfBV32 = arrayFmgr.makeArray("array_32", indexType32, elementType32); + + assertTrue(context.getFormulaManager().getFormulaType(arrayOfBV8).isArrayType()); + assertTrue(context.getFormulaManager().getFormulaType(arrayOfBV32).isArrayType()); + } + + @Test + public void BooleanSimpleFormulaAndToString() { + BooleanFormulaManager boolFmgr = context.getFormulaManager().getBooleanFormulaManager(); + + createBooleanVariables(); + + BooleanFormula pAndNotq = boolFmgr.and(p, boolFmgr.not(q)); + BooleanFormula notOf_pAndNotq = boolFmgr.not(pAndNotq); + + assertEquals("p", p.toString().trim()); + assertEquals("q", q.toString().trim()); + assertEquals(boolFmgr.not(pAndNotq).toString(), notOf_pAndNotq.toString()); + + // System.out.println("p gives: " + p.toString()); + // System.out.println("q gives: " + q.toString()); + // System.out.println("pAndNotq gives: " + pAndNotq.toString()); + + } + + @Ignore + @Test + public void BitVectorSimpleFormulaAndToString() throws Exception { + BitvectorFormulaManager bvFmgr = context.getFormulaManager().getBitvectorFormulaManager(); + createBitVectorVariables(); + + BitvectorFormula twoPlusTwo = bvFmgr.add(two, two); + BitvectorFormula four = bvFmgr.makeBitvector(4, 4); + + // BooleanFormula equalityOf4 = bvFmgr.equal(twoPlusTwo, four); + // BooleanFormula unEqualityOf4 = bvFmgr.equal(twoPlusTwo, twenty); + + // System.out.println("\"twoPlusTwo\" gives: " + twoPlusTwo.toString()); + // System.out.println("\"four \" gives: " + four.toString()); + + // // x + x + // BitvectorFormula xPlusx = bvFmgr.add(x, x); + // // x + y + // BitvectorFormula xPlusy = bvFmgr.add(x, y); + // // 2*x + // BitvectorFormula xTimes2 = bvFmgr.multiply(x, two); + // // x + x = 2*x + // BooleanFormula equality = bvFmgr.equal(xPlusx, xTimes2); + // // x + x = 2 + // BooleanFormula badEquality = bvFmgr.equal(xPlusx, two); + // + // System.out.println("x gives: " + x.toString()); + // System.out.println("y gives: " + y.toString()); + // System.out.println("two gives: " + two.toString()); + // System.out.println("twenty gives: " + twenty.toString()); + // + // System.out.println("xPlusx gives: " + xPlusx.toString()); + // System.out.println("xPlusy gives: " + xPlusy.toString()); + // System.out.println("xTimes2 gives: " + xTimes2.toString()); + // + // System.out.println("equality gives: " + equality.toString()); + // System.out.println("badEquality gives: " + badEquality.toString()); + + } + + @Test + public void BitVectorFormulaHashCode() { + BitvectorFormulaManager bvFmgr = context.getFormulaManager().getBitvectorFormulaManager(); + createBitVectorVariables(); + + BitvectorFormula anotherTwo = bvFmgr.makeBitvector(8, 2); + BitvectorFormula four = bvFmgr.makeBitvector(8, 4); + BitvectorFormula anotherFour = bvFmgr.makeBitvector(8, 4); + + assertEquals(two.toString(), anotherTwo.toString()); + assertEquals(four.toString(), anotherFour.toString()); + assertFalse("anotherTwo and two are the same objects", anotherTwo.equals(two)); + assertFalse("anotherFour and four are the same objects", anotherFour.equals(four)); + } + + // @Ignore + // @Test + public void ArraySimpleFormulaAndToString() { + ArrayFormulaManager arrayFmgr = context.getFormulaManager().getArrayFormulaManager(); + BitvectorFormulaManager bvFmgr = context.getFormulaManager().getBitvectorFormulaManager(); + + createArrayVariables(); + createBitVectorVariables(); + + BitvectorFormula one = bvFmgr.makeBitvector(8, 1); + BitvectorFormula three = bvFmgr.makeBitvector(8, 3); + BitvectorFormula four = bvFmgr.makeBitvector(8, 4); + + // arrayOfBV32 + + arrayFmgr.store(arrayOfBV8, zero, four); // TODO fix Error + arrayFmgr.store(arrayOfBV8, one, three); // TODO fix Error + + System.out.println("arrayOfBV8 : " + arrayOfBV8.toString()); + // assertEquals("p", p.toString().trim()); + // assertEquals("q", q.toString().trim()); + // assertEquals(boolFmgr.not(pAndNotq).toString(), notOf_pAndNotq.toString()); + + // System.out.println("p gives: " + p.toString()); + // System.out.println("q gives: " + q.toString()); + // System.out.println("pAndNotq gives: " + pAndNotq.toString()); + + } + + @Test + public void ProofBooleanFormula() throws InterruptedException, SolverException { + BooleanFormulaManager boolFmgr = context.getFormulaManager().getBooleanFormulaManager(); + + // create atoms + BooleanFormula xL = boolFmgr.makeVariable("xL"); + BooleanFormula xH = boolFmgr.makeVariable("xH"); + BooleanFormula yL = boolFmgr.makeVariable("yL"); + BooleanFormula yH = boolFmgr.makeVariable("yH"); + + // create formula + BooleanFormula lowXOR = boolFmgr.xor(xL, yL); + BooleanFormula highXOR = boolFmgr.xor(xH, yH); + BooleanFormula twoBitAdder = boolFmgr.and(lowXOR, highXOR); // Formula to solve + + // try (ProverEnvironment prover = + // context.newProverEnvironment(ProverOptions.GENERATE_MODELS)) { + ProverEnvironment prover = context.newProverEnvironment(ProverOptions.GENERATE_MODELS); + boolean isUnsat; + + prover.push(); + prover.addConstraint(twoBitAdder); + prover.push(); + + isUnsat = prover.isUnsat(); + assert !isUnsat; + // try (Model model = prover.getModel()) { + System.out.println("SAT : 2-bit Adder "); + // } + // } + + // prover.close(); + } + + // test getModel on formulae (Bool, BV, Array) + + // Test Prover + +} diff --git a/src/org/sosy_lab/java_smt/solvers/stp/package-info.java b/src/org/sosy_lab/java_smt/solvers/stp/package-info.java new file mode 100644 index 0000000000..54a5701193 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/package-info.java @@ -0,0 +1,2 @@ +/** */ +package org.sosy_lab.java_smt.solvers.stp; diff --git a/src/org/sosy_lab/java_smt/solvers/stp/runBasicSamples.sh b/src/org/sosy_lab/java_smt/solvers/stp/runBasicSamples.sh new file mode 100644 index 0000000000..6ac9c91a74 --- /dev/null +++ b/src/org/sosy_lab/java_smt/solvers/stp/runBasicSamples.sh @@ -0,0 +1,81 @@ +#!/bin/bash + +#------------------------------------------------------------------------------ +# check Java version +#------------------------------------------------------------------------------ + +[ -z "$JAVA" ] && JAVA=java +java_version="`$JAVA -XX:-UsePerfData -Xmx5m -version 2>&1`" +result=$? +if [ $result -eq 127 ]; then + echo "Java not found, please install Java 1.8 or newer." 1>&2 + echo "For Ubuntu: sudo apt-get install openjdk-8-jre" 1>&2 + echo "If you have installed Java 8, but it is not in your PATH," 1>&2 + echo "let the environment variable JAVA point to the \"java\" binary." 1>&2 + exit 1 +fi +if [ $result -ne 0 ]; then + echo "Failed to execute Java VM, return code was $result and output was" + echo "$java_version" + echo "Please make sure you are able to execute Java processes by running \"$JAVA\"." + exit 1 +fi +java_version="`echo "$java_version" | grep -e "^\(java\|openjdk\) version" | cut -f2 -d\\\" | sed 's/\.//g' | cut -b1-2`" +if [ -z "$java_version" ] || [ "$java_version" -lt 18 -a "$java_version" -gt 13 ] ; then + echo "Your Java version is too old, please install Java 1.8 or newer." 1>&2 + echo "For Ubuntu: sudo apt-get install openjdk-8-jre" 1>&2 + echo "If you have installed Java 8, but it is not in your PATH," 1>&2 + echo "let the environment variable JAVA point to the \"java\" binary." 1>&2 + exit 1 +fi + + +#------------------------------------------------------------------------------ +# build classpath for JavaSMT +#------------------------------------------------------------------------------ + +platform="`uname -s`" + +# where the project directory is, relative to the location of this script +case "$platform" in + Linux|CYGWIN*) + SCRIPT="$(readlink -f "$0")" + [ -n "$PATH_TO_JAVASMT" ] || PATH_TO_JAVASMT="$(readlink -f "$(dirname "$SCRIPT")")" + ;; + # other platforms like Mac don't support readlink -f + *) + [ -n "$PATH_TO_JAVASMT" ] || PATH_TO_JAVASMT="$(dirname "$0")" + ;; +esac + +if [ ! -e "$PATH_TO_JAVASMT/bin/org/sosy_lab/java_smt/example/AllSatExample.class" ] ; then + if [ ! -e "$PATH_TO_JAVASMT/javasmt.jar" ] ; then + echo "Could not find JAVASMT binary, please check path to project directory" 1>&2 + exit 1 + fi +fi + +case "$platform" in + CYGWIN*) + JAVA_VM_ARGUMENTS="$JAVA_VM_ARGUMENTS -classpath `cygpath -wp $CLASSPATH`" + ;; +esac + +export CLASSPATH="$CLASSPATH:$PATH_TO_JAVASMT/bin:$PATH_TO_JAVASMT/JAVASMT.jar:$PATH_TO_JAVASMT/lib/*:$PATH_TO_JAVASMT/lib/java/runtime/*" + +# Run Examples for Java-SMT. +# PerfDisableSharedMem avoids hsperfdata in /tmp (disable it to connect easily with VisualConsole and Co.). + +# for EXAMPLE in AllSatExample HoudiniApp Interpolation OptimizationFormulaWeights OptimizationIntReal; do + +for EXAMPLE in AllSatExample BasicSample1; do + echo "####################################################" + echo "# executing example $EXAMPLE" + echo "####################################################" + "$JAVA" \ + -XX:+PerfDisableSharedMem \ + -Djava.awt.headless=true \ + -ea \ + $JAVA_VM_ARGUMENTS \ + org.sosy_lab.java_smt.example.$EXAMPLE +done diff --git a/src/org/sosy_lab/java_smt/test/BooleanFormulaManagerTest.java b/src/org/sosy_lab/java_smt/test/BooleanFormulaManagerTest.java index 6aafbe2924..89981d747c 100644 --- a/src/org/sosy_lab/java_smt/test/BooleanFormulaManagerTest.java +++ b/src/org/sosy_lab/java_smt/test/BooleanFormulaManagerTest.java @@ -39,7 +39,8 @@ public class BooleanFormulaManagerTest extends SolverBasedTest0 { @Parameters(name = "{0}") public static Object[] getAllSolvers() { - return Solvers.values(); + // return Solvers.values(); + return new Object[] {Solvers.STP}; } @Parameter(0) diff --git a/src/org/sosy_lab/java_smt/test/BooleanFormulaSubjectTest.java b/src/org/sosy_lab/java_smt/test/BooleanFormulaSubjectTest.java index 7a832460c5..b8be12c2c9 100644 --- a/src/org/sosy_lab/java_smt/test/BooleanFormulaSubjectTest.java +++ b/src/org/sosy_lab/java_smt/test/BooleanFormulaSubjectTest.java @@ -41,7 +41,8 @@ public class BooleanFormulaSubjectTest extends SolverBasedTest0 { @Parameters(name = "{0}") public static Object[] getAllSolvers() { - return Solvers.values(); + // return Solvers.values(); + return new Object[] {Solvers.STP}; } @Parameter public Solvers solver;