Skip to content

Commit 49b606f

Browse files
committed
fix(opt): Only perform optimization when building a model
The optimization module is supposed to help us build an optimized model, so it doesn't make much sense to start optimizing before we start actually looking for a model. Previously, we were kind of forced into it because it would be incorrect to perform case splits before optimizing. Now that optimization is integrated directly into the SAT solver, it is fully independent from case splits, and this restriction does not apply any longer. This fixes issues where we would try to optimize eagerly in a small solution space and would end up enumerating the solution space before performing examining some decisions that would prune it for us, which is exactly what happened in OCamlPro#1222. Separate optimization (`do_optimize`) from case splitting (`do_case_split`) at the `Theory` level, and perform optimization in `compute_concrete_model`, i.e. at the time we switch to model generation (note: this limits the impact of optimization on unsat problems). Also change the order of decisions to consider optimized splits last for consistency, although that should not have much impact in practice.
1 parent 8728bf7 commit 49b606f

File tree

3 files changed

+66
-49
lines changed

3 files changed

+66
-49
lines changed

src/lib/reasoners/satml.ml

+46-30
Original file line numberDiff line numberDiff line change
@@ -1796,37 +1796,36 @@ module Make (Th : Theory.S) : SAT_ML with type th = Th.t = struct
17961796
make_decision env atom
17971797

17981798
and pick_branch_lit env =
1799-
match env.next_optimized_split with
1800-
| Some (fn, value, atom) ->
1801-
env.next_optimized_split <- None;
1802-
let v = atom.var in
1803-
if v.level >= 0 then (
1804-
assert (v.pa.is_true || v.na.is_true);
1805-
if atom.is_true then
1806-
env.tenv <- Th.add_objective env.tenv fn value;
1807-
pick_branch_lit env
1808-
) else (
1809-
make_decision env atom;
1810-
if atom.is_true then
1811-
env.tenv <- Th.add_objective env.tenv fn value
1812-
)
1813-
| None ->
1814-
match env.next_decisions with
1815-
| atom :: tl ->
1816-
env.next_decisions <- tl;
1799+
match env.next_decisions with
1800+
| atom :: tl ->
1801+
env.next_decisions <- tl;
1802+
pick_branch_aux env atom
1803+
| [] ->
1804+
match env.next_split with
1805+
| Some atom ->
1806+
env.next_split <- None;
18171807
pick_branch_aux env atom
1818-
| [] ->
1819-
match env.next_split with
1820-
| Some atom ->
1821-
env.next_split <- None;
1822-
pick_branch_aux env atom
1823-
| None ->
1824-
match Vheap.pop_min env.order with
1825-
| v -> pick_branch_aux env v.na
1826-
| exception Not_found ->
1827-
if Options.get_cdcl_tableaux_inst () then
1828-
assert (Matoms.is_empty env.lazy_cnf);
1829-
raise_notrace Sat
1808+
| None ->
1809+
match Vheap.pop_min env.order with
1810+
| v -> pick_branch_aux env v.na
1811+
| exception Not_found ->
1812+
if Options.get_cdcl_tableaux_inst () then
1813+
assert (Matoms.is_empty env.lazy_cnf);
1814+
match env.next_optimized_split with
1815+
| Some (fn, value, atom) ->
1816+
env.next_optimized_split <- None;
1817+
let v = atom.var in
1818+
if v.level >= 0 then (
1819+
assert (v.pa.is_true || v.na.is_true);
1820+
if atom.is_true then
1821+
env.tenv <- Th.add_objective env.tenv fn value;
1822+
pick_branch_lit env
1823+
) else (
1824+
make_decision env atom;
1825+
if atom.is_true then
1826+
env.tenv <- Th.add_objective env.tenv fn value
1827+
)
1828+
| None -> raise Sat
18301829

18311830
let pick_branch_lit env =
18321831
if env.next_dec_guard < Vec.size env.increm_guards then
@@ -1941,6 +1940,23 @@ module Make (Th : Theory.S) : SAT_ML with type th = Th.t = struct
19411940
conflict_analyze_and_fix env (C_theory ex);
19421941
compute_concrete_model ~declared_ids env
19431942

1943+
let compute_concrete_model ~declared_ids env =
1944+
assert (is_sat env);
1945+
1946+
(* Make sure all objectives are optimized before starting model
1947+
generation. This can cause us to backtrack some of the splits
1948+
done at the theory level, which is fine, because we don't care
1949+
about these at the SAT level. *)
1950+
let rec loop env =
1951+
let acts = theory_slice env in
1952+
env.tenv <- Th.do_optimize ~acts env.tenv;
1953+
if not (is_sat env) then
1954+
try solve env; assert false
1955+
with Sat -> loop env
1956+
else
1957+
compute_concrete_model ~declared_ids env
1958+
in loop env
1959+
19441960
exception Trivial
19451961

19461962
let partition atoms init =

src/lib/reasoners/theory.ml

+18-19
Original file line numberDiff line numberDiff line change
@@ -57,6 +57,8 @@ module type S = sig
5757
val extract_ground_terms : t -> Expr.Set.t
5858
val get_real_env : t -> Ccx.Main.t
5959
val get_case_split_env : t -> Ccx.Main.t
60+
val do_optimize :
61+
acts:Shostak.Literal.t Th_util.acts -> t -> t
6062
val do_case_split :
6163
?acts:Shostak.Literal.t Th_util.acts ->
6264
t -> Util.case_split_policy -> t * Expr.Set.t
@@ -670,33 +672,29 @@ module Main_Default : S = struct
670672
else
671673
[], t
672674

673-
let do_optimize acts objectives t =
675+
let do_optimize ~acts t =
676+
let objectives = t.objectives in
674677
match Objective.Model.next_unknown objectives with
675678
| Some obj ->
676679
let add_objective = acts.Th_util.acts_add_objective in
677680
optimize_obj ~for_model:false add_objective obj t;
681+
t
682+
| None -> t
683+
684+
let do_sat_splits acts t =
685+
let splits, t = sat_splits t in
686+
match splits with
687+
| [] -> do_case_split_aux t ~for_model:false
688+
| (lview, _, _) :: _ ->
689+
let lit = Shostak.(Literal.make @@ LSem (L.make lview)) in
690+
acts.Th_util.acts_add_split lit;
678691
t, SE.empty
679-
| None ->
680-
let splits, t = sat_splits t in
681-
match splits with
682-
| [] ->
683-
do_case_split_aux t ~for_model:false
684-
| (lview, _, _) :: _ ->
685-
let lit = Shostak.(Literal.make @@ LSem (L.make lview)) in
686-
acts.Th_util.acts_add_split lit;
687-
t, SE.empty
688-
689-
let do_case_split_or_optimize ?acts t =
690-
match acts with
691-
| Some acts ->
692-
do_optimize acts t.objectives t
693-
| None ->
694-
do_case_split_aux t ~for_model:false
695-
696692

697693
let do_case_split ?acts t origin =
698694
if Options.get_case_split_policy () == origin then
699-
do_case_split_or_optimize ?acts t
695+
match acts with
696+
| Some acts -> do_sat_splits acts t
697+
| None -> do_case_split_aux t ~for_model:false
700698
else
701699
t, SE.empty
702700

@@ -965,6 +963,7 @@ module Main_Empty : S = struct
965963

966964
let get_real_env _ = CC_X.empty
967965
let get_case_split_env _ = CC_X.empty
966+
let do_optimize ~acts:_ env = env
968967
let do_case_split ?acts:_ env _ = env, E.Set.empty
969968
let add_term env _ ~add_in_cs:_ = env
970969
let compute_concrete_model ~acts:_ _env = ()

src/lib/reasoners/theory.mli

+2
Original file line numberDiff line numberDiff line change
@@ -48,6 +48,8 @@ module type S = sig
4848
val extract_ground_terms : t -> Expr.Set.t
4949
val get_real_env : t -> Ccx.Main.t
5050
val get_case_split_env : t -> Ccx.Main.t
51+
val do_optimize :
52+
acts:Shostak.Literal.t Th_util.acts -> t -> t
5153
val do_case_split :
5254
?acts:Shostak.Literal.t Th_util.acts ->
5355
t -> Util.case_split_policy -> t * Expr.Set.t

0 commit comments

Comments
 (0)