diff --git a/divine/mc/machine.hpp b/divine/mc/machine.hpp index d0ed8448b3ef6b9461409c3f2b035d4a3cd3da08..e66ddf3cca7632868fd640638d8d37b7f26913b7 100644 --- a/divine/mc/machine.hpp +++ b/divine/mc/machine.hpp @@ -29,14 +29,16 @@ #include <divine/vm/setup.hpp> #include <divine/vm/eval.tpp> // FIXME +#include <brick-compose> + #include <set> #include <memory> #include <random> namespace divine::mc { - using namespace std::literals; + namespace ctx = vm::ctx; using BC = std::shared_ptr< BitCode >; using HT = brq::concurrent_hash_set< Snapshot >; @@ -124,179 +126,224 @@ namespace divine::mc namespace divine::mc::machine { - template< typename Solver, bool randomize = false > - struct Tree : TQ::Skel + struct with_pools { - using Hasher = mc::Hasher< Solver >; - using PointerV = vm::value::Pointer; - using Context = MContext; - using Eval = vm::Eval< Context >; - using Env = std::vector< std::string >; using Pool = vm::CowHeap::Pool; - auto &program() { return _bc->program(); } - auto &debug() { return _bc->debug(); } - auto &heap() { return context().heap(); } - Context &context() { return _ctx; } - - Solver _solver; - BC _bc; - Context _ctx; - std::mt19937 rand; - Pool _snap_pool, _state_pool; brick::mem::RefPool< Pool, uint8_t > _snap_refcnt; - std::shared_ptr< std::atomic< int64_t > > _total_instructions; + with_pools() : _snap_refcnt( _snap_pool ) {} - void update_instructions() + void queue_choices( TQ &tq, Snapshot snap, Origin origin, vm::State state, int take, int total ) { - *_total_instructions += context().instruction_count(); - context().instruction_count( 0 ); + for ( int i = 0; i < total; ++i ) + if ( i != take ) + { + _snap_refcnt.get( snap ); + tq.add< Task::Choose >( snap, origin, state, i, total ); + } } - Tree( BC bc ) - : Tree( bc, Context() ) + int select_choice( TQ &tq, Snapshot snap, Origin origin, vm::State state, int count ) { - _ctx.program( bc->program() ); + queue_choices( tq, snap, origin, state, 0, count ); + return 0; } + }; - Tree( BC bc, const Context &ctx ) - : _bc( bc ), _ctx( ctx ), _snap_refcnt( _snap_pool ), - _total_instructions( new std::atomic< int64_t >( 0 ) ) + struct with_counters + { + std::shared_ptr< std::atomic< int64_t > > _total_instructions; + + with_counters() + : _total_instructions( new std::atomic< int64_t >( 0 ) ) {} + }; - ~Tree() - { - ASSERT_EQ( _snap_pool.stats().total.count.used, 0 ); - update_instructions(); - } + struct with_bc + { + using Env = std::vector< std::string >; + BC _bc; - void choose( TQ &tq, Origin origin ) - { - auto snap = context().snapshot( _snap_pool ); - auto state = context()._state; + void bc( BC bc ) { _bc = bc; } + auto &program() { return _bc->program(); } + auto &debug() { return _bc->debug(); } + }; - _snap_refcnt.get( snap ); + template< typename solver_ > + struct with_solver + { + using solver_t = solver_; + solver_t _solver; + solver_t &solver(); + }; - Eval eval( context() ); - context()._choice_take = context()._choice_count = 0; - eval.refresh(); - eval.dispatch(); + template< typename solver, typename next > + struct base_ : TQ::Skel, with_pools, with_counters, with_bc, with_solver< solver >, next + { + void boot( TQ & ) {} + }; + + template< typename next > + struct select_random_ : next + { + std::mt19937 rand; + int select_choice( TQ &tq, Snapshot snap, Origin origin, vm::State state, int count ) + { using dist = std::uniform_int_distribution< int >; - if constexpr ( randomize ) - context()._choice_take = dist( 0, context()._choice_count - 1 )( rand ); + int c = dist( 0, count - 1 )( rand ); + this->queue_choices( tq, snap, origin, state, c, count ); + } + }; - /* queue up all choices other than the selected */ - for ( int i = 0; i < context()._choice_count; ++i ) - if ( i != context()._choice_take ) - { - _snap_refcnt.get( snap ); - tq.add< Task::Choose >( snap, origin, state, i, context()._choice_count ); - } + template< typename ctx, typename next > + struct with_context_ : next + { + using context_t = ctx; + ctx _ctx; + ctx &context() { return _ctx; } + auto &heap() { return context().heap(); } - compute( tq, origin, snap ); + void update_instructions() + { + *this->_total_instructions += context().instruction_count(); + context().instruction_count( 0 ); } - virtual std::pair< State, bool > store() + ~with_context_() { - return { State( context().snapshot( _state_pool ) ), true }; + update_instructions(); } + }; - virtual bool loop_closed( Origin & ) + template< typename next > + struct compute_ : next + { + using Eval = vm::Eval< typename next::context_t >; + + void eval_choice( TQ &tq, Origin origin ) { - return false; + auto snap = this->context().snapshot( this->_snap_pool ); + auto state = this->context()._state; + + this->_snap_refcnt.get( snap ); + + Eval eval( this->context() ); + this->context()._choice_take = this->context()._choice_count = 0; + eval.refresh(); + eval.dispatch(); + + this->context()._choice_take = + this->select_choice( tq, snap, origin, state, this->context()._choice_count ); + compute( tq, origin, snap ); } - void schedule( TQ &tq, Origin origin, Snapshot cont_from ) + void eval_interrupt( TQ &tq, Origin origin, Snapshot cont_from ) { - if ( context().frame().null() ) + if ( this->context().frame().null() ) { Label lbl; - lbl.accepting = context().flags_any( _VM_CF_Accepting ); - lbl.error = context().flags_any( _VM_CF_Error ); - auto [ state, isnew ] = store(); + lbl.accepting = this->context().flags_any( _VM_CF_Accepting ); + lbl.error = this->context().flags_any( _VM_CF_Error ); + auto [ state, isnew ] = this->store(); tq.add< StateTQ::Skel::Edge >( State( origin.snap ), state, lbl, isnew ); } - else if ( !loop_closed( origin ) ) + else if ( this->loop_closed( origin ) ) + { + /* TODO: check for error/accepting flags */ + } + else return compute( tq, origin, cont_from ); } bool feasible() { - if ( context().flags_any( _VM_CF_Cancel ) ) + if ( this->context().flags_any( _VM_CF_Cancel ) ) return false; - if ( context()._assume.null() ) + if ( this->context()._assume.null() ) return true; - // TODO validate the path condition - return _solver.feasible( context().heap(), context()._assume ); + return this->_solver.feasible( this->context().heap(), this->context()._assume ); } void compute( TQ &tq, Origin origin, Snapshot cont_from = Snapshot() ) { - auto destroy = [&]( auto p ) { heap().snap_put( _snap_pool, p, false ); }; - + auto destroy = [&]( auto p ) { this->heap().snap_put( this->_snap_pool, p, false ); }; auto cleanup = [&] { if ( cont_from ) - _snap_refcnt.put( cont_from, destroy ); + this->_snap_refcnt.put( cont_from, destroy ); }; brick::types::Defer _( cleanup ); - Eval eval( context() ); + Eval eval( this->context() ); bool choice = eval.run_seq( !!cont_from ); if ( !feasible() ) return; if ( choice ) - choose( tq, origin ); + eval_choice( tq, origin ); else - schedule( tq, origin, cont_from ); - + eval_interrupt( tq, origin, cont_from ); } - virtual void boot( TQ &tq ) + bool boot( TQ &tq ) { - Eval eval( context() ); - vm::setup::boot( context() ); + this->context().program( this->program() ); + Eval eval( this->context() ); + vm::setup::boot( this->context() ); eval.run(); - } + next::boot( tq ); - using TQ::Skel::run; + if ( vm::setup::postboot_check( this->context() ) ) + return tq.add< Task::Schedule >( this->store().first ), true; + else + return false; + } - void run( TQ &tq, Task::Boot ) + void schedule( TQ &tq, Task::Schedule e ) { - boot( tq ); + TRACE( "task (schedule)", e ); + this->context().load( this->_state_pool, e.from.snap ); + vm::setup::scheduler( this->context() ); + compute( tq, Origin( e.from.snap ) ); + } - if ( vm::setup::postboot_check( context() ) ) - tq.add< Task::Schedule >( store().first ); + void choose( TQ &tq, Task::Choose c ) + { + TRACE( "task (choose)", c ); + if ( !c.total ) + return; + this->context().load( this->_snap_pool, c.snap ); + this->context()._state = c.state; + this->context()._choice_take = c.choice; + this->context()._choice_count = c.total; + this->context().flush_ptr2i(); + this->context().load_pc(); + compute( tq, c.origin, c.snap ); } + }; - void run( TQ &tq, Task::Schedule e ) + template< typename next > + struct tree_store_ : next + { + virtual std::pair< State, bool > store() { - context().load( _state_pool, e.from.snap ); - vm::setup::scheduler( context() ); - compute( tq, Origin( e.from.snap ) ); + return { State( this->context().snapshot( this->_state_pool ) ), true }; } - void run( TQ &tq, Task::Choose c ) + virtual bool loop_closed( Origin & ) { - context().load( _snap_pool, c.snap ); - context()._state = c.state; - context()._choice_take = c.choice; - context()._choice_count = c.total; - context().flush_ptr2i(); - context().load_pc(); - compute( tq, c.origin, c.snap ); + return false; } auto make_hasher() { - Hasher h( _state_pool, context().heap(), this->_solver ); - h._root = context().state_ptr(); + Hasher h( this->_state_pool, this->context().heap(), this->_solver ); + h._root = this->context().state_ptr(); ASSERT( !h._root.null() ); return h; } @@ -305,69 +352,100 @@ namespace divine::mc::machine { return make_hasher().equal_symbolic( a, b ); } + + ~tree_store_() + { + ASSERT_EQ( this->_snap_pool.stats().total.count.used, 0 ); + } + }; - template< typename Solver > - struct Graph : Tree< Solver > + template< typename next > + struct graph_store_ : next { - using Hasher = mc::Hasher< Solver >; - using Tree< Solver >::context; + using Hasher = mc::Hasher< typename next::solver_t >; Hasher _hasher; - struct Ext - { - HT ht_sched; - bool overwrite = false; - } _ext; - - Graph( BC bc ) : Tree< Solver >( bc ), - _hasher( this->_state_pool, context().heap(), this->_solver ) - {} + HT _ht_sched; auto &hasher() { return _hasher; } - void enable_overwrite() { _ext.hasher.overwrite = true; } bool equal( Snapshot a, Snapshot b ) { return hasher().equal_symbolic( a, b ); } + graph_store_() : _hasher( this->_state_pool, this->heap(), this->_solver ) {} std::pair< State, bool > store( HT &table ) { - auto snap = context().snapshot( this->_state_pool ); + auto snap = this->context().snapshot( this->_state_pool ); auto r = table.insert( snap, hasher() ); if ( r->load() != snap ) { ASSERT( !_hasher.overwrite ); ASSERT( !r.isnew() ); - context().heap().snap_put( this->_state_pool, snap ); + this->heap().snap_put( this->_state_pool, snap ); } return { State( *r ), r.isnew() }; } - std::pair< State, bool > store() override + std::pair< State, bool > store() { - return store( _ext.ht_sched ); + return store( _ht_sched ); } - bool loop_closed( Origin &origin ) override + bool loop_closed( Origin &origin ) { auto [ state, isnew ] = store( origin.ht_loop ); if ( isnew ) - context().flush_ptr2i(); + this->context().flush_ptr2i(); else - context().load( this->_state_pool, state.snap ); + { + this->context().load( this->_state_pool, state.snap ); + TRACE( "loop closed at", state.snap ); + } + return !isnew; } - virtual void boot( TQ &tq ) override + void boot( TQ &tq ) { - Tree< Solver >::boot( tq ); - hasher()._root = context().state_ptr(); + next::boot( tq ); + hasher()._root = this->context().state_ptr(); } }; + + template< typename next > + struct dispatch_ : next + { + using next::run; + + void run( TQ &tq, Task::Boot ) { next::boot( tq ); } + void run( TQ &tq, Task::Schedule e ) { next::schedule( tq, e ); } + void run( TQ &tq, Task::Choose c ) { next::choose( tq, c ); } + }; + + template< typename solver > + using base = brq::module< base_, solver >; + + template< typename ctx > + using with_context = brq::module< with_context_, ctx >; + + using compute = brq::module< compute_ >; + using select_random = brq::module< select_random_ >; + using tree_store = brq::module< tree_store_ >; + using graph_store = brq::module< graph_store_ >; + using dispatch = brq::module< dispatch_ >; + + template< typename solver > + using tree = brq::compose_stack< dispatch, compute, tree_store, + with_context< MContext >, base< solver > >; + + template< typename solver > + using graph = brq::compose_stack< dispatch, compute, graph_store, with_context< MContext >, + base< solver > >; } namespace divine::mc { - using TMachine = machine::Tree< smt::NoSolver >; - using GMachine = machine::Graph< smt::NoSolver >; + using TMachine = machine::tree< smt::NoSolver >; + using GMachine = machine::graph< smt::NoSolver >; template< typename M > auto weave( M &machine ) diff --git a/divine/mc/t-machine.hpp b/divine/mc/t-machine.hpp index fb0a876719b9860184a41f30a24d52f693328883..d7e455f21abd76a5ae463875030d790ac9137665 100644 --- a/divine/mc/t-machine.hpp +++ b/divine/mc/t-machine.hpp @@ -85,18 +85,22 @@ namespace divine::t_mc TEST( instance ) { auto bc = prog( "void __boot( void *e ) { __vm_ctl_flag( 0, 0b10000 ); }" ); - mc::TMachine m( bc ); + mc::TMachine m; + m.bc( bc ); } using Search = mc::Search< mc::TQ >; using Expand = mc::TQ::Tasks::Expand; using Edge = mc::TQ::Tasks::Edge; + auto tmachine( mc::BC bc ) { mc::TMachine tm; tm.bc( bc ); return tm; } + auto gmachine( mc::BC bc ) { mc::GMachine tm; tm.bc( bc ); return tm; } + TEST( simple ) { bool found = false; auto state = [&]( Expand ) { found = true; }; - auto machine = mc::TMachine( prog_int( 4, "x - 1" ) ); + auto machine = tmachine( prog_int( 4, "x - 1" ) ); mc::weave( machine ).observe( state ).start(); ASSERT( found ); } @@ -104,7 +108,7 @@ namespace divine::t_mc TEST( hasher ) { bool ran = false; - auto machine = mc::GMachine( prog_int( 4, "x - 1" ) ); + auto machine = gmachine( prog_int( 4, "x - 1" ) ); auto check = [&]( Expand s ) { ASSERT( machine.hasher()._pool.size( s.from.snap ) ); @@ -117,7 +121,7 @@ namespace divine::t_mc TEST( start_twice ) { - auto machine = mc::TMachine( prog_int( 4, "x - 1" ) ); + auto machine = tmachine( prog_int( 4, "x - 1" ) ); mc::State i1, i2; mc::weave( machine ).observe( [&]( Expand e ) { i1 = e.from; } ).start(); @@ -130,7 +134,7 @@ namespace divine::t_mc { mc::State i1, i2; - auto m1 = mc::TMachine( prog_int( 4, "x - 1" ) ), m2 = m1; + auto m1 = tmachine( prog_int( 4, "x - 1" ) ), m2 = m1; mc::weave( m1 ).observe( [&]( Expand e ) { i1 = e.from; } ).start(); mc::weave( m2 ).observe( [&]( Expand e ) { i2 = e.from; } ).start(); @@ -141,7 +145,7 @@ namespace divine::t_mc TEST( unfold ) { std::vector< mc::Snapshot > snaps; - auto machine = mc::TMachine( prog_int( 4, "x - 1" ) ); + auto machine = tmachine( prog_int( 4, "x - 1" ) ); auto state = [&]( Expand e ) { @@ -157,7 +161,8 @@ namespace divine::t_mc template< typename M > void _search( std::shared_ptr< mc::BitCode > bc, int sc, int ec ) { - M machine( bc ); + M machine; + machine.bc( bc ); int edgecount = 0, statecount = 0; auto edge = [&]( Edge ) { ++edgecount; }; auto state = [&]( Expand ) { ++statecount; };