Skip to content

Commit 19f9207

Browse files
committedDec 26, 2023
Use polymorphic variant for mode
1 parent 9142602 commit 19f9207

File tree

5 files changed

+42
-51
lines changed

5 files changed

+42
-51
lines changed
 

‎src/kcas/kcas.ml

+13-20
Original file line numberDiff line numberDiff line change
@@ -136,7 +136,9 @@ let[@inline] resume_awaiters = function
136136
| [ awaiter ] -> resume_awaiter awaiter
137137
| awaiters -> List.iter resume_awaiter awaiters
138138

139-
type mode = Lock_free | Obstruction_free
139+
module Mode = struct
140+
type t = [ `Lock_free | `Obstruction_free ]
141+
end
140142

141143
type 'a state = {
142144
mutable before : 'a;
@@ -164,7 +166,7 @@ and _ tdt =
164166
This field must be first, see [root_as_atomic] and
165167
[tree_as_ref]. *)
166168
timeout : [ `Set | `Unset ] Timeout.t;
167-
mutable mode : mode;
169+
mutable mode : Mode.t;
168170
mutable validate_counter : int;
169171
mutable post_commit : Action.t;
170172
}
@@ -231,13 +233,6 @@ let[@inline] is_determined = function
231233
| R After | R Before -> true
232234
end
233235

234-
module Mode = struct
235-
type t = mode
236-
237-
let lock_free = Lock_free
238-
let obstruction_free = Obstruction_free
239-
end
240-
241236
let[@inline] isnt_int x = not (Obj.is_int (Obj.repr x))
242237

243238
let[@inline] rec release_after_rec which = function
@@ -545,20 +540,18 @@ let dec x = x - 1
545540
module Loc = struct
546541
type 'a t = 'a loc
547542

548-
let make ?(padded = false) ?(mode = Mode.obstruction_free) after =
543+
let make ?(padded = false) ?(mode = `Obstruction_free) after =
549544
let state = new_state after
550-
and id =
551-
if mode == Mode.obstruction_free then Id.nat_id () else Id.neg_id ()
552-
in
545+
and id = if mode == `Obstruction_free then Id.nat_id () else Id.neg_id () in
553546
make_loc padded state id
554547

555548
let make_contended ?mode after = make ~padded:true ?mode after
556549

557-
let make_array ?(padded = false) ?(mode = Mode.obstruction_free) n after =
550+
let make_array ?(padded = false) ?(mode = `Obstruction_free) n after =
558551
assert (0 <= n);
559552
let state = new_state after
560553
and id =
561-
(if mode == Mode.obstruction_free then Id.nat_ids n else Id.neg_ids n)
554+
(if mode == `Obstruction_free then Id.nat_ids n else Id.neg_ids n)
562555
- (n - 1)
563556
in
564557
Array.init n @@ fun i -> make_loc padded state (id + i)
@@ -584,7 +577,7 @@ module Loc = struct
584577
get_as (Timeout.alloc_opt timeoutf) f loc (atomic_get (as_atomic loc))
585578

586579
let[@inline] get_mode loc =
587-
if loc.id < 0 then Mode.lock_free else Mode.obstruction_free
580+
if loc.id < 0 then `Lock_free else `Obstruction_free
588581

589582
let compare_and_set ?(backoff = Backoff.default) loc before after =
590583
let state = new_state after in
@@ -660,7 +653,7 @@ module Xt = struct
660653

661654
let[@inline] is_obstruction_free (Xt xt_r : _ t) loc =
662655
(* Fenceless is safe as we are accessing a private location. *)
663-
xt_r.mode == Mode.obstruction_free && 0 <= loc.id
656+
xt_r.mode == `Obstruction_free && 0 <= loc.id
664657

665658
let[@inline] update_new loc f xt lt gt =
666659
(* Fenceless is safe inside transactions as each log update has a fence. *)
@@ -895,9 +888,9 @@ module Xt = struct
895888
if a_cmp_followed_by_a_cas < status then begin
896889
if finish xt root (verify xt root) then success xt result
897890
else begin
898-
(* We switch to [Mode.lock_free] as there was
891+
(* We switch to [`Lock_free] as there was
899892
interference. *)
900-
commit_once_alloc backoff Mode.lock_free xt tx
893+
commit_once_alloc backoff `Lock_free xt tx
901894
end
902895
end
903896
else if
@@ -965,7 +958,7 @@ module Xt = struct
965958
commit backoff xt tx
966959

967960
let[@inline] commit ?timeoutf ?(backoff = Backoff.default)
968-
?(mode = Mode.obstruction_free) { tx } =
961+
?(mode = `Obstruction_free) { tx } =
969962
let timeout = Timeout.alloc_opt timeoutf
970963
and rot = U Leaf
971964
and validate_counter = initial_validate_period

‎src/kcas/kcas.mli

+16-16
Original file line numberDiff line numberDiff line change
@@ -140,18 +140,17 @@ end
140140

141141
(** Operating modes of the [k-CAS-n-CMP] algorithm. *)
142142
module Mode : sig
143-
type t
144-
(** Type of an operating mode of the [k-CAS-n-CMP] algorithm. *)
145-
146-
val lock_free : t
147-
(** In [lock_free] mode the algorithm makes sure that at least one domain will
143+
type t =
144+
[ `Lock_free
145+
(** In [`Lock_free] mode the algorithm makes sure that at least one domain will
148146
be able to make progress at the cost of performing read-only operations as
149147
read-write operations. *)
150-
151-
val obstruction_free : t
152-
(** In [obstruction_free] mode the algorithm proceeds optimistically and
148+
| `Obstruction_free
149+
(** In [`Obstruction_free] mode the algorithm proceeds optimistically and
153150
allows read-only operations to fail due to interference from other domains
154-
that might have been prevented in the {!lock_free} mode. *)
151+
that might have been prevented in the [`Lock_free] mode. *)
152+
]
153+
(** Type of an operating mode of the [k-CAS-n-CMP] algorithm. *)
155154
end
156155

157156
(** {1 Individual locations}
@@ -185,8 +184,8 @@ module Loc : sig
185184
of using more memory. It is not recommended to use [~padded:true] for
186185
short lived shared memory locations.
187186
188-
The optional [mode] argument defaults to {!Mode.obstruction_free}. If
189-
explicitly specified as {!Mode.lock_free}, the location will always be
187+
The optional {{!Mode.t} [mode]} argument defaults to [`Obstruction_free].
188+
If explicitly specified as [`Lock_free], the location will always be
190189
accessed using the lock-free operating mode. This may improve performance
191190
in rare cases where a location is updated frequently and obstruction-free
192191
read-only accesses would almost certainly suffer from interference. *)
@@ -288,8 +287,9 @@ end
288287
done internally by the library implementation. Only logically invisible
289288
writes to shared memory locations are performed during this phase.
290289
291-
3. In {!Mode.obstruction_free} a third phase verifies all read-only
292-
operations. This is also done internally by the library implementation.
290+
3. In [`Obstruction_free] {{!Mode.t} mode} a third phase verifies all
291+
read-only operations. This is also done internally by the library
292+
implementation.
293293
294294
Each phase may fail. In particular, in the first phase, as no changes to
295295
shared memory have yet been attempted, it is safe, for example, to raise
@@ -537,9 +537,9 @@ module Xt : sig
537537
{!Retry.Invalid} to explicitly request a retry or any other exception to
538538
abort the transaction.
539539
540-
The default for [commit] is {!Mode.obstruction_free}. However, after
541-
enough attempts have failed during the verification step, [commit]
542-
switches to {!Mode.lock_free}. *)
540+
The default {{!Mode.t} [mode]} for [commit] is [`Obstruction_free].
541+
However, after enough attempts have failed during the verification step,
542+
[commit] switches to [`Lock_free]. *)
543543

544544
(**/**)
545545

‎src/kcas_data/accumulator.ml

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@ let make ?n_way n =
1111
| None -> n_way_default
1212
| Some n_way -> n_way |> Int.min n_way_max |> Bits.ceil_pow_2
1313
in
14-
let a = Loc.make_array ~padded:true ~mode:Mode.lock_free n_way 0 in
14+
let a = Loc.make_array ~padded:true ~mode:`Lock_free n_way 0 in
1515
Loc.set (Array.unsafe_get a 0) n;
1616
a
1717

‎test/kcas/loc_modes.ml

+3-3
Original file line numberDiff line numberDiff line change
@@ -5,9 +5,9 @@ let loop_count = try int_of_string Sys.argv.(1) with _ -> Util.iter_factor
55
let mode =
66
Some
77
(try
8-
if Sys.argv.(2) = "obstruction-free" then Mode.obstruction_free
9-
else Mode.lock_free
10-
with _ -> Mode.lock_free)
8+
if Sys.argv.(2) = "obstruction-free" then `Obstruction_free
9+
else `Lock_free
10+
with _ -> `Lock_free)
1111

1212
(* Number of shared counters being used to try to cause interference *)
1313

‎test/kcas/test.ml

+9-11
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@ let run_domains = function
4444
(* *)
4545

4646
let test_non_linearizable_xt () =
47-
[ Mode.obstruction_free; Mode.lock_free ]
47+
[ `Obstruction_free; `Lock_free ]
4848
|> List.iter @@ fun mode ->
4949
let barrier = Barrier.make 2
5050
and n_iter = 100 * Util.iter_factor
@@ -63,7 +63,7 @@ let test_non_linearizable_xt () =
6363
in
6464

6565
let atomically tx =
66-
if Random.bool () then Xt.commit ~mode:Mode.obstruction_free tx
66+
if Random.bool () then Xt.commit ~mode:`Obstruction_free tx
6767
else Xt.commit tx
6868
in
6969

@@ -103,7 +103,7 @@ let test_set () =
103103
(* *)
104104

105105
let test_no_skew_xt () =
106-
[ Mode.obstruction_free; Mode.lock_free ]
106+
[ `Obstruction_free; `Lock_free ]
107107
|> List.iter @@ fun mode ->
108108
let barrier = Barrier.make 3 in
109109
let test_finished = Atomic.make false in
@@ -168,7 +168,7 @@ let test_no_skew_xt () =
168168
(* *)
169169

170170
let test_get_seq_xt () =
171-
[ Mode.obstruction_free; Mode.lock_free ]
171+
[ `Obstruction_free; `Lock_free ]
172172
|> List.iter @@ fun mode ->
173173
let barrier = Barrier.make 4 in
174174
let test_finished = Atomic.make false in
@@ -225,7 +225,7 @@ let test_get_seq_xt () =
225225
(* *)
226226

227227
let test_stress_xt n nb_loop =
228-
[ Mode.obstruction_free; Mode.lock_free ]
228+
[ `Obstruction_free; `Lock_free ]
229229
|> List.iter @@ fun mode ->
230230
let make_loc n =
231231
let rec loop n out =
@@ -326,7 +326,7 @@ let test_post_commit () =
326326
begin
327327
try
328328
count := 0;
329-
Xt.commit ~mode:Mode.obstruction_free { tx }
329+
Xt.commit ~mode:`Obstruction_free { tx }
330330
with Exit -> ()
331331
end;
332332
assert (!count = expect)
@@ -613,11 +613,9 @@ let test_timeout () =
613613
(* *)
614614

615615
let test_mode () =
616-
assert (Loc.get_mode (Loc.make ~mode:Mode.lock_free 0) == Mode.lock_free);
617-
assert (
618-
Loc.get_mode (Loc.make ~mode:Mode.obstruction_free 0)
619-
== Mode.obstruction_free);
620-
assert (Loc.get_mode (Loc.make 0) == Mode.obstruction_free)
616+
assert (Loc.get_mode (Loc.make ~mode:`Lock_free 0) == `Lock_free);
617+
assert (Loc.get_mode (Loc.make ~mode:`Obstruction_free 0) == `Obstruction_free);
618+
assert (Loc.get_mode (Loc.make 0) == `Obstruction_free)
621619

622620
(* *)
623621

0 commit comments

Comments
 (0)
Please sign in to comment.