Safe Haskell | Safe-Inferred |
---|---|
Language | Haskell2010 |
This module implements an algorithm described in the "Component-based Synthesis Applied to Bitvector Programs" paper.
https://www.microsoft.com/en-us/research/wp-content/uploads/2010/02/bv.pdf
Given a program specification along with a library of available components it synthesizes an actual program using an off-the-shelf SMT-solver.
The specification is an arbitrary datatype that is an instance of the SynthSpec
class.
The library is a list of SynthComponent
instances.
There are three entry points to this module: standardExAllProcedure
, refinedExAllProcedure
and exAllProcedure
.
Synopsis
- module Data.SBV.Program.Types
- module Data.SBV.Program.SimpleLibrary
- module Data.SBV.Program.Utils
- standardExAllProcedure :: forall a comp spec. (SymVal a, Show a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> spec a -> IO (Either SynthesisError (Program Location (comp a)))
- refinedExAllProcedure :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> spec a -> IO (Either SynthesisError (Program Location (comp a)))
- exAllProcedure :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> spec a -> IO (Either SynthesisError (Program Location (comp a)))
- createProgramLocs :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> Word -> Symbolic (Program SLocation (comp a))
- constrainLocs :: (SynthSpec spec a, SynthComponent comp spec a) => Word -> Word -> Program SLocation (comp a) -> Symbolic ()
- createProgramVarsWith :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => (String -> Symbolic (SBV a)) -> [comp a] -> Word -> Symbolic (Program (SBV a) (comp a))
- createVarsConstraints :: SynthComponent comp spec a => Program SLocation (comp a) -> Program (SBV a) (comp a) -> (SBool, SBool)
- createConstantVars :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> Symbolic (Program (SBV a) (comp a))
- combineProgramVars :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => Program (SBV a) (comp a) -> Program (SBV a) (comp a) -> Program (SBV a) (comp a)
- instructionGetValue :: forall a comp spec m. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => Instruction SLocation (comp a) -> Instruction (SBV a) (comp a) -> Query (Instruction Location (comp a))
Definitions for specification and component classes and the resulting data type
module Data.SBV.Program.Types
Predefined components library
Various utility functions
module Data.SBV.Program.Utils
Package entry points
standardExAllProcedure Source #
:: forall a comp spec. (SymVal a, Show a, SynthSpec spec a, SynthComponent comp spec a) | |
=> [comp a] | Component library |
-> spec a | Specification of program being synthesized |
-> IO (Either SynthesisError (Program Location (comp a))) |
An implementation of StandardExAllSolver presented in section 6.1 of the paper. As stated in the paper, this implementation boils down to exhaustive enumeration of possible solutions, and as such isn't effective. It can be used to better understand how the synthesis procedure works and provides a lot of debugging output. It also doesn't support constant components. Do not use this procedure for solving real problems.
refinedExAllProcedure Source #
:: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) | |
=> [comp a] | Component library |
-> spec a | Specification of program being synthesized |
-> IO (Either SynthesisError (Program Location (comp a))) |
An implementation of RefinedExAllSolver presented in section 6.2 of the paper.
This is an improved version of standardExAllProcedure
. It only keeps input
values \(|\vec I|\) in \(S\) and uses different synthesis constraints on synthesis
and verification steps.
:: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) | |
=> [comp a] | Component library |
-> spec a | Specification of program being synthesized |
-> IO (Either SynthesisError (Program Location (comp a))) |
This procedure is not part of the paper. It uses forall quantification directly when creating variables from the \(T\) set. As consequence it requires an SMT-solver than can handle foralls (for instance, Z3). This procedure is the easiest to understand.
Auxiliary functions that make up synthesis procedures
createProgramLocs :: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => [comp a] -> Word -> Symbolic (Program SLocation (comp a)) Source #
First step of each synthesis procedure. Given a library of components and a number of program's inputs, it creates existentially quantified location variables (members of set \(L\) in the paper) for each component and for the program itself.
:: (SynthSpec spec a, SynthComponent comp spec a) | |
=> Word | The \(M\) constant from the paper, which equals to \(N + |\vec I|\), where N is the size of the library. |
-> Word | Number of program inputs \(|\vec I|\). |
-> Program SLocation (comp a) | |
-> Symbolic () |
Second step of each synthesis procedure. It applies constraints on location variables from section 5 of the original paper. These constraints include well-formedness constraint \(ψ_{wfp}\), acyclicity constraint \(ψ_{acyc}\) and consistency constraint \(ψ_{cons}\). Constraints are not returned from this function, but are applied immediately. Section 5 of the paper also talks about connectivity constraint \(ψ_{conn}\), which is not created here.
createProgramVarsWith Source #
:: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) | |
=> (String -> Symbolic (SBV a)) | |
-> [comp a] | Component library. |
-> Word | Number of program inputs \(|\vec I|\). |
-> Symbolic (Program (SBV a) (comp a)) |
Third step of the synthesis process. It creates variables that represent
actual inputs/outputs values (members of the set \(T\) in the paper). This
function resembles createProgramLocs
, but unlike it allows creating both existentially
and universally quantified variables. Standard and Refined procedures pass
sbvExists
to create existentially quantified variables, while exAllProcedure
uses sbvForall
.
createVarsConstraints :: SynthComponent comp spec a => Program SLocation (comp a) -> Program (SBV a) (comp a) -> (SBool, SBool) Source #
Last building block of the synthesis process. This function creates \(ψ_{conn}\) and \(φ_{lib}\) constraints and return them.
Constant components handling
:: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) | |
=> [comp a] | Component library. |
-> Symbolic (Program (SBV a) (comp a)) |
Special version of createProgramVarsWith
for constant components.
A constant component is a component having specArity
\(=0\). The original
paper slightly touches this topic in the last paragraph of section 7.
This function always uses existential quantification and only operates on
constant components. The Program
returned from this function contains
undefined
values for programIOs
and non-constant programInstructions
.
The user is expected to call createProgramVarsWith
later and then use
combineProgramVars
to merge two results.
:: forall a comp spec. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) | |
=> Program (SBV a) (comp a) | The result of |
-> Program (SBV a) (comp a) | The result of |
-> Program (SBV a) (comp a) |
instructionGetValue :: forall a comp spec m. (SymVal a, SynthSpec spec a, SynthComponent comp spec a) => Instruction SLocation (comp a) -> Instruction (SBV a) (comp a) -> Query (Instruction Location (comp a)) Source #
Smart version of getValue
for Instruction
.
For each component it gets solutions for location variable, effectively turning
'Instruction SLocation (comp a)' into 'Instruction Location (comp a)'.
For constant components it additionaly fills 'comp a' part of the structure
with its returning value.