Panda is an extension for Java Pathfinder [1], which introduces support for predicate abstraction and various other abstractions of numeric data domains, such as signs and intervals. The name "Panda" stands for Predicate Abstraction in Dynamic Analysis.
It started as a Google Summer of Code (GSoC) project in 2012 [2] with the goal to implement basic abstractions of numeric data domains. Support for predicate abstraction was added in the scope of another GSoC project in 2013 [3]. Currently, we focus on automated generation of predicates, modeling Java collections, and support for concurrency (multiple threads).
-
Java 7
To be able to run Panda, it is necessary to have JDK 7 installed on your system. You can download the JDK 7 directly from the Oracle web site. -
CVC4
Currently only the Linux x86-64 binaries of CVC4 are included within the project repository. For platforms other than Linux x86-64, you also need to obtain the appropriate binary at the CVC4 web site. -
Z3
Currently only the Linux x86-64 binaries of Z3 are included within the project repository. For platforms other than Linux x86-64 or in case of trouble with dynamically linked libraries, you also need to obtain the appropriate binary or to compile the solver from source. Both precompiled binaries and source code can be found at the Z3 web site.
The installation guide is mostly specific to Linux/Unix. Modify the commands appropriately for other platforms (Windows, Mac OS).
Create an empty directory (e.g. ~/workspace
) where you would like to install the project
mkdir -p ~/workspace
Obtain the latest Panda
git clone https://github.com/d3sformal/panda.git
Build it using Ant
cd ~/workspace/panda
ant clean build
Make sure that bin/mathsat
is executable on your platform before continuing.
To perform verification of a target program, it is necessary to provide a configuration file (.jpf
). Assuming that the main class of a target program is target.Target
and that it is stored in src/examples
, then the content of the file Target.jpf
(typically in the same directory) would look like:
@using=jpf-abstraction
target=target.Target
classpath=${jpf-abstraction}/build/examples
sourcepath=${jpf-abstraction}/src/examples
panda.abstract_domain=PREDICATES ${jpf-abstraction}/src/examples/target/Target.pred
listener=gov.nasa.jpf.abstraction.AbstractListener
Notable configuration attributes are:
- target
The target program (Java class with themain
method). - panda.abstract_domain
The abstraction to be used - in our casePREDICATES
- followed by a path to the file with input predicates. Other supported abstractions are described here. - listener
Thegov.nasa.jpf.abstraction.AbstractListener
listener is mandatory for the predicate abstraction to work properly. It is also possible to specify additional listeners to get more verbose output:
gov.nasa.jpf.abstraction.util.PredicateValuationMonitor
prints the values of all predicates after each instruction.gov.nasa.jpf.listener.ExecTracker
orgov.nasa.jpf.abstraction.util.InstructionTracker
to put the output of other listeners into the context of executed bytecode instructions.
- vm.serializer.class
It is necessary to set this option togov.nasa.jpf.abstraction.PredicateAbstractionSerializer
to enable abstract state matching.
Additional configuration options are described on the JPF-core configuration page.
The file with input predicates (usually having the suffix .pred
) is divided into sections that we call contexts.
[static]
...
[object ...]
...
[method ...]
...
In static contexts, it is possible to define predicates over static fields and numeric constants.
In object contexts, it is possible to define predicates that refer also to object fields. The scope is defined by the type of objects which is specified in the header of the section (e.g. [object target.Target]
).
In method contexts, it is possible to define predicates that refer also to local variables. The scope is once again determined by the method full name supplied in the header of the section (e.g. [method target.Target.main]
).
The number and order of occurrences of the sections is not limited in any way.
Each of the sections may contain an arbitrary number of predicates.
The predicates must be defined in the form of equalities and inequalities of arithmetic expressions over numeric constants and access expressions.
For example:
a.b = c + d - 1
An access expression can be a local variable, a static field, an object field, or array element access.
o
o.f
a[0]
class(pkg.Class).f
alength(arrlen, a)
Here, the class(...)
symbol distinguishes a static field access from an object field access. The expression wrapped in class(...)
must refer to a Java class name. The alength(arrlen, ...)
symbol is a special accessor for array length that distinguishes it from a field access. These special symbols are needed because the predicate language is not typed and has no information about runtime classes.
There are two notations for specifying the access expressions:
- Java-like dot notation demonstrated above
- Function notation:
sfread(f, pkg.Class)
forclass(pkg.Class).f
fread(f, o)
foro.f
aread(arr, a, i)
fora[i]
In a method context, you may define predicates over the keyword return
, which are used for propagation of truth values of predicates over method call boundaries.
To run Panda, simply issue the following command within the directory containing it.
bin/run.sh <path-to-a-jpf-file>
The following example can be found in the project repository and can be run using the command mentioned below. The example consists of three files.
package arraylength;
public class ALength {
public static void main(String[] args) {
int i = 3;
int a[] = new int[i];
}
}
[method arraylength.ALength.main]
alength(arrlen, a) = -1
alength(arrlen, a) = i
alength(arrlen, a) = 3
i = 3
@using=jpf-abstraction
target=arraylength.ALength
classpath=${jpf-abstraction}/build/examples
sourcepath=${jpf-abstraction}/src/examples
panda.abstract_domain=PREDICATES ${jpf-abstraction}/src/examples/arraylength/ALength.pred
listener=gov.nasa.jpf.abstraction.AbstractListener
For the purpose of this example, we omitted some of the non-essential listeners, which print only debugging information.
To run the example, simply issue the following command within the directory containing Panda.
bin/run.sh src/examples/arraylength/ALength.jpf
The output of the command should look like this:
Running Panda ...
[method arraylength.ALength.main]
alength(arrlen, a) = -1
alength(arrlen, a) = i
alength(arrlen, a) = 3
i = 3
JavaPathfinder v7.0 (rev 1117) - (C) RIACS/NASA Ames Research Center
====================================================== system under test
arraylength.ALength.main()
====================================================== search started: 20/09/13 16:39
====================================================== results
no errors detected
====================================================== statistics
elapsed time: 00:00:00
states: new=1, visited=0, backtracked=1, end=1
search: maxDepth=1, constraints hit=0
choice generators: thread=1 (signal=0, lock=1, shared ref=0), data=0
heap: new=349, released=12, max live=0, gc-cycles=1
instructions: 3276
max memory: 120MB
loaded code: classes=56, methods=1112
====================================================== search finished: 20/09/13 16:39
Panda parses the input file and prints all the collected predicates in the default function notation. If some other listeners were added, then a lot of output may follow. In the end, there is the expected statement no errors detected
and statistics provided by JPF-core (number of choices, etc).