mirror of
				https://github.com/Z3Prover/z3
				synced 2025-10-26 17:29:21 +00:00 
			
		
		
		
	iterative deepening experiment (no PQ yet). the hardness heuristic is still naive!
This commit is contained in:
		
							parent
							
								
									9ea6f2ea86
								
							
						
					
					
						commit
						1fae0dea16
					
				
					 3 changed files with 62 additions and 8 deletions
				
			
		|  | @ -15,4 +15,5 @@ def_module_params('smt_parallel', | ||||||
|                         ('num_split_lits', UINT, 2, 'how many literals, k, we split on to create 2^k cubes'), |                         ('num_split_lits', UINT, 2, 'how many literals, k, we split on to create 2^k cubes'), | ||||||
|                         ('depth_splitting_only', BOOL, False, 'only apply frugal cube strategy, and only on deepest (biggest) cubes from the batch manager'), |                         ('depth_splitting_only', BOOL, False, 'only apply frugal cube strategy, and only on deepest (biggest) cubes from the batch manager'), | ||||||
|                         ('backbone_detection', BOOL, False, 'apply backbone literal heuristic'), |                         ('backbone_detection', BOOL, False, 'apply backbone literal heuristic'), | ||||||
|  |                         ('iterative_deepening', BOOL, False, 'deepen cubes based on iterative hardness cutoff heuristic'), | ||||||
|               )) |               )) | ||||||
|  |  | ||||||
|  | @ -75,7 +75,47 @@ namespace smt { | ||||||
|                             vector<expr_ref_vector> returned_cubes; |                             vector<expr_ref_vector> returned_cubes; | ||||||
|                             returned_cubes.push_back(cube);  |                             returned_cubes.push_back(cube);  | ||||||
|                             auto split_atoms = get_split_atoms(); |                             auto split_atoms = get_split_atoms(); | ||||||
|  | 
 | ||||||
|  |                             if (m_config.m_iterative_deepening) { | ||||||
|  |                                 unsigned conflicts_before    = ctx->m_stats.m_num_conflicts; | ||||||
|  |                                 unsigned propagations_before = ctx->m_stats.m_num_propagations + ctx->m_stats.m_num_bin_propagations; | ||||||
|  |                                 unsigned decisions_before    = ctx->m_stats.m_num_decisions; | ||||||
|  |                                 unsigned restarts_before     = ctx->m_stats.m_num_restarts; | ||||||
|  | 
 | ||||||
|  |                                 lbool r = check_cube(cube); | ||||||
|  | 
 | ||||||
|  |                                 unsigned conflicts_after    = ctx->m_stats.m_num_conflicts; | ||||||
|  |                                 unsigned propagations_after = ctx->m_stats.m_num_propagations + ctx->m_stats.m_num_bin_propagations; | ||||||
|  |                                 unsigned decisions_after    = ctx->m_stats.m_num_decisions; | ||||||
|  |                                 unsigned restarts_after     = ctx->m_stats.m_num_restarts; | ||||||
|  | 
 | ||||||
|  |                                 // per-cube deltas
 | ||||||
|  |                                 unsigned conflicts_delta    = conflicts_after - conflicts_before; | ||||||
|  |                                 unsigned propagations_delta = propagations_after - propagations_before; | ||||||
|  |                                 unsigned decisions_delta    = decisions_after - decisions_before; | ||||||
|  |                                 unsigned restarts_delta     = restarts_after - restarts_before; | ||||||
|  |                                 LOG_WORKER(1, " cube deltas: conflicts: " << conflicts_delta << " propagations: " << propagations_delta << " decisions: " << decisions_delta << " restarts: " << restarts_delta << "\n"); | ||||||
|  | 
 | ||||||
|  |                                 // tuned experimentally
 | ||||||
|  |                                 const double w_conflicts = 1.0; | ||||||
|  |                                 const double w_propagations = 0.001; | ||||||
|  |                                 const double w_decisions = 0.1; | ||||||
|  |                                 const double w_restarts = 0.5; | ||||||
|  | 
 | ||||||
|  |                                 const double cube_hardness = w_conflicts * conflicts_delta + | ||||||
|  |                                     w_propagations * propagations_delta + | ||||||
|  |                                     w_decisions * decisions_delta + | ||||||
|  |                                     w_restarts * restarts_delta; | ||||||
|  | 
 | ||||||
|  |                                 const double avg_hardness = b.update_avg_cube_hardness(cube_hardness); | ||||||
|  |                                 const double factor = 1;  // can tune for multiple of avg hardness later
 | ||||||
|  |                                 bool should_split = cube_hardness >= avg_hardness * factor; | ||||||
|  |                                 LOG_WORKER(1, " cube hardness: " << cube_hardness << " avg: " << avg_hardness << " should-split: " << should_split << "\n"); | ||||||
|  |                                 if (should_split) | ||||||
|                                     b.return_cubes(m_l2g, returned_cubes, split_atoms); |                                     b.return_cubes(m_l2g, returned_cubes, split_atoms); | ||||||
|  |                             } else { | ||||||
|  |                                 b.return_cubes(m_l2g, returned_cubes, split_atoms); | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                         if (m_config.m_backbone_detection) { |                         if (m_config.m_backbone_detection) { | ||||||
|                             expr_ref_vector backbone_candidates = find_backbone_candidates(); |                             expr_ref_vector backbone_candidates = find_backbone_candidates(); | ||||||
|  | @ -152,6 +192,7 @@ namespace smt { | ||||||
|         m_config.m_max_greedy_cubes = pp.max_greedy_cubes(); |         m_config.m_max_greedy_cubes = pp.max_greedy_cubes(); | ||||||
|         m_config.m_num_split_lits = pp.num_split_lits(); |         m_config.m_num_split_lits = pp.num_split_lits(); | ||||||
|         m_config.m_backbone_detection = pp.backbone_detection(); |         m_config.m_backbone_detection = pp.backbone_detection(); | ||||||
|  |         m_config.m_iterative_deepening = pp.iterative_deepening(); | ||||||
| 
 | 
 | ||||||
|         // don't share initial units
 |         // don't share initial units
 | ||||||
|         ctx->pop_to_base_lvl(); |         ctx->pop_to_base_lvl(); | ||||||
|  | @ -294,7 +335,7 @@ namespace smt { | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         for (unsigned i = 0; i < std::min(m_max_batch_size / p.num_threads, (unsigned)m_cubes.size()) && !m_cubes.empty(); ++i) { |         for (unsigned i = 0; i < std::min(m_max_batch_size / p.num_threads, (unsigned)m_cubes.size()) && !m_cubes.empty(); ++i) { | ||||||
|             if (m_config.m_depth_splitting_only) { |             if (m_config.m_depth_splitting_only || m_config.m_iterative_deepening) { | ||||||
|                 // get the deepest set of cubes
 |                 // get the deepest set of cubes
 | ||||||
|                 auto& deepest_cubes = m_cubes_depth_sets.rbegin()->second; |                 auto& deepest_cubes = m_cubes_depth_sets.rbegin()->second; | ||||||
|                 unsigned idx = rand() % deepest_cubes.size(); |                 unsigned idx = rand() % deepest_cubes.size(); | ||||||
|  | @ -447,8 +488,6 @@ namespace smt { | ||||||
|     ------------------------------------------------------------------------------------------------------------------------------------------------------------ |     ------------------------------------------------------------------------------------------------------------------------------------------------------------ | ||||||
|     Final thought (do this!): use greedy strategy by a policy when C_batch, A_batch, A_worker are "small". -- want to do this. switch to frugal strategy after reaching size limit |     Final thought (do this!): use greedy strategy by a policy when C_batch, A_batch, A_worker are "small". -- want to do this. switch to frugal strategy after reaching size limit | ||||||
|     */ |     */ | ||||||
| 
 |  | ||||||
|     // currenly, the code just implements the greedy strategy
 |  | ||||||
|     void parallel::batch_manager::return_cubes(ast_translation& l2g, vector<expr_ref_vector>const& C_worker, expr_ref_vector const& A_worker) { |     void parallel::batch_manager::return_cubes(ast_translation& l2g, vector<expr_ref_vector>const& C_worker, expr_ref_vector const& A_worker) { | ||||||
| //        SASSERT(!m_config.never_cube());
 | //        SASSERT(!m_config.never_cube());
 | ||||||
| 
 | 
 | ||||||
|  | @ -500,7 +539,7 @@ namespace smt { | ||||||
| 
 | 
 | ||||||
|         std::scoped_lock lock(mux); |         std::scoped_lock lock(mux); | ||||||
|         unsigned max_greedy_cubes = 1000; |         unsigned max_greedy_cubes = 1000; | ||||||
|         bool greedy_mode = (m_cubes.size() <= max_greedy_cubes) && !m_config.m_frugal_cube_only; |         bool greedy_mode = (m_cubes.size() <= max_greedy_cubes) && !m_config.m_frugal_cube_only && !m_config.m_iterative_deepening; // for iterative deepening, our hardness metric is cube-specific, so it only makes sense for frugal approach for now
 | ||||||
|         unsigned a_worker_start_idx = 0; |         unsigned a_worker_start_idx = 0; | ||||||
| 
 | 
 | ||||||
|         //
 |         //
 | ||||||
|  | @ -535,7 +574,7 @@ namespace smt { | ||||||
|                 g_cube.push_back(l2g(atom)); |                 g_cube.push_back(l2g(atom)); | ||||||
|             unsigned start = m_cubes.size(); // update start after adding each cube so we only process the current cube being added
 |             unsigned start = m_cubes.size(); // update start after adding each cube so we only process the current cube being added
 | ||||||
| 
 | 
 | ||||||
|             if (m_config.m_depth_splitting_only) { |             if (m_config.m_depth_splitting_only || m_config.m_iterative_deepening) { | ||||||
|                 // need to add the depth set if it doesn't exist yet
 |                 // need to add the depth set if it doesn't exist yet
 | ||||||
|                 if (m_cubes_depth_sets.find(g_cube.size()) == m_cubes_depth_sets.end()) { |                 if (m_cubes_depth_sets.find(g_cube.size()) == m_cubes_depth_sets.end()) { | ||||||
|                     m_cubes_depth_sets[g_cube.size()] = vector<expr_ref_vector>(); |                     m_cubes_depth_sets[g_cube.size()] = vector<expr_ref_vector>(); | ||||||
|  | @ -571,7 +610,7 @@ namespace smt { | ||||||
|                 expr_ref g_atom(l2g(A_worker[i]), l2g.to()); |                 expr_ref g_atom(l2g(A_worker[i]), l2g.to()); | ||||||
|                 if (!m_split_atoms.contains(g_atom)) |                 if (!m_split_atoms.contains(g_atom)) | ||||||
|                     m_split_atoms.push_back(g_atom); |                     m_split_atoms.push_back(g_atom); | ||||||
|                 if (m_config.m_depth_splitting_only) { |                 if (m_config.m_depth_splitting_only || m_config.m_iterative_deepening) { | ||||||
|                     add_split_atom_deepest_cubes(g_atom); |                     add_split_atom_deepest_cubes(g_atom); | ||||||
|                 } else { |                 } else { | ||||||
|                     add_split_atom(g_atom, initial_m_cubes_size); |                     add_split_atom(g_atom, initial_m_cubes_size); | ||||||
|  | @ -755,6 +794,7 @@ namespace smt { | ||||||
|         m_config.m_frugal_cube_only = sp.frugal_cube_only(); |         m_config.m_frugal_cube_only = sp.frugal_cube_only(); | ||||||
|         m_config.m_never_cube = sp.never_cube(); |         m_config.m_never_cube = sp.never_cube(); | ||||||
|         m_config.m_depth_splitting_only = sp.depth_splitting_only(); |         m_config.m_depth_splitting_only = sp.depth_splitting_only(); | ||||||
|  |         m_config.m_iterative_deepening = sp.iterative_deepening(); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     void parallel::batch_manager::collect_statistics(::statistics& st) const { |     void parallel::batch_manager::collect_statistics(::statistics& st) const { | ||||||
|  |  | ||||||
|  | @ -51,6 +51,7 @@ namespace smt { | ||||||
|                 bool m_frugal_cube_only = false; |                 bool m_frugal_cube_only = false; | ||||||
|                 bool m_never_cube = false;  |                 bool m_never_cube = false;  | ||||||
|                 bool m_depth_splitting_only = false; |                 bool m_depth_splitting_only = false; | ||||||
|  |                 bool m_iterative_deepening = false; | ||||||
|             }; |             }; | ||||||
|             struct stats { |             struct stats { | ||||||
|                 unsigned m_max_cube_depth = 0; |                 unsigned m_max_cube_depth = 0; | ||||||
|  | @ -65,6 +66,8 @@ namespace smt { | ||||||
|             stats m_stats; |             stats m_stats; | ||||||
|             expr_ref_vector m_split_atoms; // atoms to split on
 |             expr_ref_vector m_split_atoms; // atoms to split on
 | ||||||
|             vector<expr_ref_vector> m_cubes; |             vector<expr_ref_vector> m_cubes; | ||||||
|  |             updatable_priority_queue::priority_queue<expr_ref_vector, double> m_cubes_pq; | ||||||
|  |              | ||||||
|             std::map<unsigned, vector<expr_ref_vector>> m_cubes_depth_sets; // map<vec<cube>> contains sets of cubes, key is depth/size of cubes in the set
 |             std::map<unsigned, vector<expr_ref_vector>> m_cubes_depth_sets; // map<vec<cube>> contains sets of cubes, key is depth/size of cubes in the set
 | ||||||
|             unsigned m_max_batch_size = 10; |             unsigned m_max_batch_size = 10; | ||||||
|             unsigned m_exception_code = 0; |             unsigned m_exception_code = 0; | ||||||
|  | @ -73,6 +76,9 @@ namespace smt { | ||||||
|             obj_hashtable<expr> shared_clause_set; // for duplicate filtering on per-thread clause expressions
 |             obj_hashtable<expr> shared_clause_set; // for duplicate filtering on per-thread clause expressions
 | ||||||
|             vector<parameter_state> m_parameters_state; |             vector<parameter_state> m_parameters_state; | ||||||
| 
 | 
 | ||||||
|  |             double m_avg_cube_hardness = 0.0; | ||||||
|  |             unsigned m_solved_cube_count = 0; | ||||||
|  | 
 | ||||||
|             // called from batch manager to cancel other workers if we've reached a verdict
 |             // called from batch manager to cancel other workers if we've reached a verdict
 | ||||||
|             void cancel_workers() { |             void cancel_workers() { | ||||||
|                 IF_VERBOSE(1, verbose_stream() << "Canceling workers\n"); |                 IF_VERBOSE(1, verbose_stream() << "Canceling workers\n"); | ||||||
|  | @ -108,6 +114,12 @@ namespace smt { | ||||||
|             void collect_clause(ast_translation& l2g, unsigned source_worker_id, expr* e); |             void collect_clause(ast_translation& l2g, unsigned source_worker_id, expr* e); | ||||||
|             expr_ref_vector return_shared_clauses(ast_translation& g2l, unsigned& worker_limit, unsigned worker_id); |             expr_ref_vector return_shared_clauses(ast_translation& g2l, unsigned& worker_limit, unsigned worker_id); | ||||||
| 
 | 
 | ||||||
|  |             double update_avg_cube_hardness(double hardness) { | ||||||
|  |                 m_avg_cube_hardness = (m_avg_cube_hardness * m_solved_cube_count + hardness) / (m_solved_cube_count + 1); | ||||||
|  |                 m_solved_cube_count++; | ||||||
|  |                 return m_avg_cube_hardness; | ||||||
|  |             } | ||||||
|  | 
 | ||||||
|             void collect_statistics(::statistics& st) const; |             void collect_statistics(::statistics& st) const; | ||||||
|             lbool get_result() const; |             lbool get_result() const; | ||||||
|         }; |         }; | ||||||
|  | @ -126,6 +138,7 @@ namespace smt { | ||||||
|                 unsigned m_max_greedy_cubes = 1000; |                 unsigned m_max_greedy_cubes = 1000; | ||||||
|                 unsigned m_num_split_lits = 2; |                 unsigned m_num_split_lits = 2; | ||||||
|                 bool m_backbone_detection = false; |                 bool m_backbone_detection = false; | ||||||
|  |                 bool m_iterative_deepening = false; | ||||||
|             }; |             }; | ||||||
| 
 | 
 | ||||||
|             unsigned id; // unique identifier for the worker
 |             unsigned id; // unique identifier for the worker
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue