diff --git a/bench/bench_spsc_queue.ml b/bench/bench_spsc_queue.ml index 7b6fc4bd..4cf06094 100644 --- a/bench/bench_spsc_queue.ml +++ b/bench/bench_spsc_queue.ml @@ -1,51 +1,18 @@ open Multicore_bench -let run_one ~unsafe ~budgetf ?(size_exponent = 3) - ?(n_msgs = 80 * Util.iter_factor) () = - let init _ = () in - let work, before = - if unsafe then - let module Queue = Saturn.Single_prod_single_cons_queue_unsafe in - let t = Queue.create ~size_exponent in +module type BENCH = sig + val run_suite : budgetf:float -> Metric.t list +end - let before () = - while Queue.size t <> 0 do - Queue.pop_exn t |> ignore - done; - let n = Random.int ((1 lsl size_exponent) + 1) in - for i = 1 to n do - Queue.push_exn t i - done - in - let work i () = - if i = 0 then - let rec loop n = - if 0 < n then - if Queue.try_push t n then loop (n - 1) - else begin - Domain.cpu_relax (); - loop n - end - in - loop n_msgs - else - let rec loop n = - if 0 < n then - match Queue.pop_opt t with - | Some _ -> loop (n - 1) - | None -> - Domain.cpu_relax (); - loop n - in - loop n_msgs - in - (work, before) - else - let module Queue = Saturn.Single_prod_single_cons_queue in +module Make (Queue : Spsc_queue_intf.SPSC_queue) : BENCH = struct + let run_one ~budgetf ?(size_exponent = 3) ?(n_msgs = 80 * Util.iter_factor) () + = + let init _ = () in + let work, before = let t = Queue.create ~size_exponent in let before () = - while Queue.size t <> 0 do + while Queue.length t <> 0 do Queue.pop_exn t |> ignore done; let n = Random.int ((1 lsl size_exponent) + 1) in @@ -76,21 +43,19 @@ let run_one ~unsafe ~budgetf ?(size_exponent = 3) loop n_msgs in (work, before) - in + in - let config = - Printf.sprintf "2 workers, capacity %d%s" (1 lsl size_exponent) - (if unsafe then " (unsafe)" else "") - in - Times.record ~budgetf ~n_domains:2 ~before ~init ~work () - |> Times.to_thruput_metrics ~n:n_msgs ~singular:"message" ~config + let config = + Printf.sprintf "2 workers, capacity %d" (1 lsl size_exponent) + in + Times.record ~budgetf ~n_domains:2 ~before ~init ~work () + |> Times.to_thruput_metrics ~n:n_msgs ~singular:"message" ~config -let run_suite ~budgetf = - let run ~unsafe = + let run_suite ~budgetf = [ 0; 3; 6; 9; 12; 15 ] |> List.concat_map @@ fun size_exponent -> - run_one ~budgetf ~size_exponent ~unsafe () - in - List.fold_right2 - (fun safe unsafe acc -> safe :: unsafe :: acc) - (run ~unsafe:false) (run ~unsafe:true) [] + run_one ~budgetf ~size_exponent () +end + +module Safe = Make (Saturn.Single_prod_single_cons_queue) +module Unsafe = Make (Saturn.Single_prod_single_cons_queue_unsafe) diff --git a/bench/dune b/bench/dune index cf68fa33..81fd3e6d 100644 --- a/bench/dune +++ b/bench/dune @@ -17,6 +17,11 @@ let () = (copy ../src/michael_scott_queue/michael_scott_queue_intf.ml michael_scott_queue_intf.ml)) (package saturn)) +(rule + (action + (copy ../src/spsc_queue/spsc_queue_intf.mli spsc_queue_intf.ml)) + (package saturn)) + (rule (action (copy ../src/bounded_queue/bounded_queue_intf.mli bounded_queue_intf.ml)) diff --git a/bench/main.ml b/bench/main.ml index 4cf7f0dd..57c83b1a 100644 --- a/bench/main.ml +++ b/bench/main.ml @@ -4,7 +4,9 @@ let benchmarks = ("Saturn Queue_unsafe", Bench_queue.Unsafe.run_suite); ("Saturn Bounded_Queue", Bench_bounded_queue.Safe.run_suite); ("Saturn Bounded_Queue_unsafe", Bench_bounded_queue.Unsafe.run_suite); - ("Saturn Single_prod_single_cons_queue", Bench_spsc_queue.run_suite); + ("Saturn Single_prod_single_cons_queue", Bench_spsc_queue.Safe.run_suite); + ( "Saturn Single_prod_single_cons_queue_unsafe", + Bench_spsc_queue.Unsafe.run_suite ); ("Saturn Size", Bench_size.run_suite); ("Saturn Skiplist", Bench_skiplist.run_suite); ("Saturn Htbl", Bench_htbl.Safe.run_suite); diff --git a/src/dune b/src/dune index fa8a6d27..c4d88b1c 100644 --- a/src/dune +++ b/src/dune @@ -12,7 +12,7 @@ let () = (library (name saturn) (public_name saturn) - (modules_without_implementation htbl_intf bounded_queue_intf) + (modules_without_implementation htbl_intf bounded_queue_intf spsc_queue_intf) (libraries backoff multicore-magic |} ^ maybe_threads ^ {| )) diff --git a/src/spsc_queue/dune b/src/spsc_queue/dune new file mode 100644 index 00000000..2dead6c5 --- /dev/null +++ b/src/spsc_queue/dune @@ -0,0 +1,8 @@ +(mdx + (package saturn) + (enabled_if + (and + (<> %{os_type} Win32) + (>= %{ocaml_version} 5.0.0))) + (libraries saturn) + (files spsc_queue_intf.mli)) diff --git a/src/spsc_queue/spsc_queue.ml b/src/spsc_queue/spsc_queue.ml index 2ef1a6bb..f62326c0 100644 --- a/src/spsc_queue/spsc_queue.ml +++ b/src/spsc_queue/spsc_queue.ml @@ -39,6 +39,9 @@ type 'a t = { } exception Full +exception Empty + +(* *) let create ~size_exponent = if size_exponent < 0 || Sys.int_size - 2 < size_exponent then @@ -52,6 +55,23 @@ let create ~size_exponent = let head_cache = Padded_int_ref.make s 0 in { array; tail; tail_cache; head; head_cache } +let of_list_exn ~size_exponent values = + if size_exponent < 0 || Sys.int_size - 2 < size_exponent then + invalid_arg "size_exponent out of range"; + let size = 1 lsl size_exponent in + let nvalues = List.length values in + if nvalues > size then raise Full; + let array = Array.make size None in + List.iteri (fun i elt -> Array.set array i (Some elt)) values; + let tail = Atomic.make_contended nvalues in + let s = Obj.size (Obj.repr tail) in + let tail_cache = Padded_int_ref.make s nvalues in + let head = Atomic.make_contended 0 in + let head_cache = Padded_int_ref.make s 0 in + { array; tail; tail_cache; head; head_cache } + +(* *) + type _ mono = Unit : unit mono | Bool : bool mono let push_as (type r) t element (mono : r mono) : r = @@ -66,7 +86,7 @@ let push_as (type r) t element (mono : r mono) : r = head == head_cache then match mono with Unit -> raise_notrace Full | Bool -> false else begin - Array.unsafe_set t.array (tail land (size - 1)) (Some element); + Array.set t.array (tail land (size - 1)) (Some element); Atomic.incr t.tail; match mono with Unit -> () | Bool -> true end @@ -74,9 +94,13 @@ let push_as (type r) t element (mono : r mono) : r = let push_exn t element = push_as t element Unit let try_push t element = push_as t element Bool -exception Empty +(* *) + +type ('a, _) poly = + | Option : ('a, 'a option) poly + | Value : ('a, 'a) poly + | Unit : ('a, unit) poly -type ('a, _) poly = Option : ('a, 'a option) poly | Value : ('a, 'a) poly type op = Peek | Pop let pop_or_peek_as (type a r) (t : a t) op (poly : (a, r) poly) : r = @@ -88,25 +112,28 @@ let pop_or_peek_as (type a r) (t : a t) op (poly : (a, r) poly) : r = let tail = Atomic.get t.tail in Padded_int_ref.set t.tail_cache tail; tail_cache == tail - then match poly with Value -> raise_notrace Empty | Option -> None + then match poly with Value | Unit -> raise_notrace Empty | Option -> None else let index = head land (Array.length t.array - 1) in - let v = Array.unsafe_get t.array index in + let v = Array.get t.array index in begin match op with | Pop -> - Array.unsafe_set t.array index None; + Array.set t.array index None; Atomic.incr t.head | Peek -> () end; - match poly with Value -> Option.get v | Option -> v + match poly with Value -> Option.get v | Option -> v | Unit -> () let pop_exn t = pop_or_peek_as t Pop Value let pop_opt t = pop_or_peek_as t Pop Option let peek_exn t = pop_or_peek_as t Peek Value let peek_opt t = pop_or_peek_as t Peek Option +let drop_exn t = pop_or_peek_as t Pop Unit + +(* *) -let size t = +let length t = let tail = Atomic.get t.tail in let head = Atomic.get t.head in tail - head diff --git a/src/spsc_queue/spsc_queue_intf.ml b/src/spsc_queue/spsc_queue_intf.ml deleted file mode 100644 index 526ac80e..00000000 --- a/src/spsc_queue/spsc_queue_intf.ml +++ /dev/null @@ -1,79 +0,0 @@ -module type SPSC_queue = sig - (** Single producer single consumer queue. *) - - type 'a t - (** Type of single-producer single-consumer non-resizable domain-safe queue - that works in FIFO order. *) - - val create : size_exponent:int -> 'a t - (** [create ~size_exponent:int] returns a new queue of maximum size - [2^size_exponent] and initially empty. *) - - val size : 'a t -> int - (** [size] returns the size of the queue. This method linearizes only when - called from either consumer or producer domain. Otherwise, it is safe to - call but provides only an *indication* of the size of the structure. *) - - (** {1 Producer functions} *) - - exception Full - (** Raised when {!push_exn} is applied to a full queue. - - This exception is meant to avoid allocations required by an option type. - As such, it does not register backtrace information and it is recommended - to use the following pattern to catch it. - - {[ - match push_exn q v with - | () -> (* ... *) - | exception Full -> (* ... *) - ]} *) - - val push_exn : 'a t -> 'a -> unit - (** [push q v] adds the element [v] at the end of the queue [q]. This method - can be used by at most one domain at a time. - - @raise Full if the queue is full. *) - - val try_push : 'a t -> 'a -> bool - (** [try_push q v] tries to add the element [v] at the end of the queue [q]. - It fails it the queue [q] is full. This method can be used by at most one - domain at a time. *) - - (** {2 Consumer functions} *) - - exception Empty - (** Raised when {!pop_exn} or {!peek_exn} is applied to an empty queue. - - This exception is meant to avoid allocations required by an option type. - As such, it does not register backtrace information and it is recommended - to use the following pattern to catch it. - - {[ - match pop_exn q with - | value -> (* ... *) - | exception Empty -> (* ... *) - ]} *) - - val pop_exn : 'a t -> 'a - (** [pop_exn q] removes and returns the first element in queue [q]. This method - can be used by at most one domain at a time. - - @raise Empty if [q] is empty. *) - - val pop_opt : 'a t -> 'a option - (** [pop_opt q] removes and returns the first element in queue [q], or returns - [None] if the queue is empty. This method can be used by at most one domain - at a time. *) - - val peek_exn : 'a t -> 'a - (** [peek_exn q] returns the first element in queue [q]. This method can be - used by at most one domain at a time. - - @raise Empty if [q] is empty. *) - - val peek_opt : 'a t -> 'a option - (** [peek_opt q] returns the first element in queue [q], or [None] if the - queue is empty. This method can be used by at most one domain at a - time. *) -end diff --git a/src/spsc_queue/spsc_queue_intf.mli b/src/spsc_queue/spsc_queue_intf.mli new file mode 100644 index 00000000..1a4853dd --- /dev/null +++ b/src/spsc_queue/spsc_queue_intf.mli @@ -0,0 +1,186 @@ +module type SPSC_queue = sig + (** Single producer single consumer queue. + + **Note**: This queue does not include safety mechanisms to prevent + misuse. If consumer-only functions are called concurrently by multiple + domains, the queue may enter an unexpected state, due to data races + and a lack of linearizability. The same goes for producer-only functions. + *) + + (** {1 API} *) + + type 'a t + (** Represents a single-producer single-consumer non-resizable queue + that works in FIFO order. *) + + val create : size_exponent:int -> 'a t + (** [create ~size_exponent] creates a new single-producer single-consumer + queue with a maximum size of [2^size_exponent] and initially empty. + + 🐌 This is a linear-time operation in [2^size_exponent]. *) + + val of_list_exn : size_exponent:int -> 'a list -> 'a t + (** [of_list_exn ~size_exponent list] creates a new queue from a list. + + @raises Full if the length of [list] is greater than [2^size_exponent]. + + 🐌 This is a linear-time operation. + + {[ + # open Saturn.Single_prod_single_cons_queue + # let t : int t = of_list_exn ~size_exponent:6 [1;2;3;4] + val t : int t = + # pop_opt t + - : int option = Some 1 + # pop_opt t + - : int option = Some 2 + ]} + *) + + val length : 'a t -> int + (** [length] returns the length of the queue. This method linearizes only when + called from either the consumer or producer domain. Otherwise, it is safe to + call but provides only an *indication* of the size of the structure. *) + + (** {2 Producer functions} *) + + exception Full + (** Raised when {!push_exn} is applied to a full queue. *) + + val push_exn : 'a t -> 'a -> unit + (** [push queue elt] adds the element [elt] at the end of the [queue]. + This method can be used by at most one domain at a time. + + @raises Full if the [queue] is full. *) + + val try_push : 'a t -> 'a -> bool + (** [try_push queue elt] tries to add the element [elt] at the end of the + [queue]. If the queue [q] is full, [false] is returned. This method can be + used by at most one domain at a time. *) + + (** {2 Consumer functions} *) + + exception Empty + (** Raised when {!pop_exn}, {!peek_exn}, or {!drop_exn} is applied to an empty + queue. *) + + val pop_exn : 'a t -> 'a + (** [pop_exn queue] removes and returns the first element in [queue]. This + method can be used by at most one domain at a time. + + @raises Empty if the [queue] is empty. *) + + val pop_opt : 'a t -> 'a option + (** [pop_opt queue] removes and returns [Some] of the first element of the + [queue], or [None] if the queue is empty. This method can be used by at most + one domain at a time. *) + + val peek_exn : 'a t -> 'a + (** [peek_exn queue] returns the first element in [queue] without removing it. + This method can be used by at most one domain at a time. + + @raises Empty if the [queue] is empty. *) + + val peek_opt : 'a t -> 'a option + (** [peek_opt queue] returns [Some] of the first element in [queue], or [None] + if the queue is empty. This method can be used by at most one domain at a + time. *) + + val drop_exn : 'a t -> unit + (** [drop_exn queue] removes the top element of the [queue]. + + @raises Empty if the [queue] is empty. *) + + (** {1 Examples} *) + + (** {2 Sequential example} *) + + (** {[ + # open Saturn.Single_prod_single_cons_queue + # let t : int t = create ~size_exponent:2 + val t : int t = + # push_exn t 1 + - : unit = () + # push_exn t 2 + - : unit = () + # try_push t 3 + - : bool = true + # try_push t 4 + - : bool = true + # try_push t 5 + - : bool = false + + # pop_opt t + - : int option = Some 1 + # peek_opt t + - : int option = Some 2 + # drop_exn t + - : unit = () + # pop_exn t + - : int = 3 + # pop_opt t + - : int option = Some 4 + # pop_exn t + Exception: Saturn__Spsc_queue.Empty. + ]} *) + + (** {2 Parallel example} + Note: The barrier is used in this example solely to make the results more + interesting by increasing the likelihood of parallelism. Spawning a domain is + a costly operation, especially compared to the relatively small amount of work + being performed here. In practice, using a barrier in this manner is unnecessary. + + {@ocaml non-deterministic=command[ + # open Saturn.Single_prod_single_cons_queue + # let t : int t = create ~size_exponent:5 + val t : int t = + + # let nwork = 5 + val nwork : int = 5 + # let barrier = Atomic.make 2 + + val barrier : int Atomic.t = + # let consumer_work () = + (* Atomic.decr barrier; + while Atomic.get barrier <> 0 do Domain.cpu_relax () done; *) + let rec loop n = + if n < 1 then () + else + (Domain.cpu_relax (); + match pop_opt t with + | Some p -> Format.printf "Popped %d\n%!" p; loop (n-1) + | None -> loop n) + in + loop nwork + val consumer_work : unit -> unit = + + # let producer_work () = + (* Atomic.decr barrier; + while Atomic.get barrier <> 0 do Domain.cpu_relax () done; *) + for i = 1 to nwork do + Domain.cpu_relax (); + try_push t i |> ignore; + Format.printf "Pushed %d\n%!" i + done + val producer_work : unit -> unit = + + # let consumer = Domain.spawn consumer_work + val consumer : unit Domain.t = + # let producer = Domain.spawn producer_work + Pushed 1 + Popped 1 + Pushed 2 + Popped 2 + Pushed 3 + Popped 3 + Pushed 4 + Popped 4 + Popped 5 + Pushed 5 + val producer : unit Domain.t = + # Domain.join consumer + - : unit = () + # Domain.join producer + - : unit = () + ]} *) +end diff --git a/src/spsc_queue/spsc_queue_unsafe.ml b/src/spsc_queue/spsc_queue_unsafe.ml index 7b0c5b10..333d7d1d 100644 --- a/src/spsc_queue/spsc_queue_unsafe.ml +++ b/src/spsc_queue/spsc_queue_unsafe.ml @@ -35,6 +35,9 @@ type 'a t = { } exception Full +exception Empty + +(* *) let create ~size_exponent = if size_exponent < 0 || Sys.int_size - 2 < size_exponent then @@ -48,6 +51,22 @@ let create ~size_exponent = { array; tail; tail_cache; head; head_cache } |> Multicore_magic.copy_as_padded +let of_list_exn ~size_exponent values = + if size_exponent < 0 || Sys.int_size - 2 < size_exponent then + invalid_arg "size_exponent out of range"; + let size = 1 lsl size_exponent in + let nvalues = List.length values in + if nvalues > size then raise Full; + let array = Array.make size (Obj.magic ()) in + List.iteri (fun i elt -> Array.unsafe_set array i (Obj.magic elt)) values; + let tail = Atomic.make_contended nvalues in + let tail_cache = ref nvalues |> Multicore_magic.copy_as_padded in + let head = Atomic.make_contended 0 in + let head_cache = ref 0 |> Multicore_magic.copy_as_padded in + { array; tail; tail_cache; head; head_cache } + +(* *) + type _ mono = Unit : unit mono | Bool : bool mono (* NOTE: Uses of [@inline never] prevent Flambda from noticing that we might be @@ -73,9 +92,13 @@ let[@inline never] push_as (type r) t element (mono : r mono) : r = let push_exn t element = push_as t element Unit let try_push t element = push_as t element Bool -exception Empty +(* *) + +type ('a, _) poly = + | Option : ('a, 'a option) poly + | Value : ('a, 'a) poly + | Unit : ('a, unit) poly -type ('a, _) poly = Option : ('a, 'a option) poly | Value : ('a, 'a) poly type op = Peek | Pop let[@inline never] pop_or_peek_as (type a r) t op (poly : (a, r) poly) : r = @@ -87,7 +110,7 @@ let[@inline never] pop_or_peek_as (type a r) t op (poly : (a, r) poly) : r = let tail = Atomic.get t.tail in t.tail_cache := tail; tail_cache == tail - then match poly with Value -> raise_notrace Empty | Option -> None + then match poly with Value | Unit -> raise_notrace Empty | Option -> None else let index = head land (Array.length t.array - 1) in let v = Array.unsafe_get t.array index |> Obj.magic in @@ -98,14 +121,15 @@ let[@inline never] pop_or_peek_as (type a r) t op (poly : (a, r) poly) : r = Atomic.incr t.head | Peek -> () end; - match poly with Value -> v | Option -> Some v + match poly with Value -> v | Option -> Some v | Unit -> () let pop_exn t = pop_or_peek_as t Pop Value let pop_opt t = pop_or_peek_as t Pop Option let peek_exn t = pop_or_peek_as t Peek Value let peek_opt t = pop_or_peek_as t Peek Option +let drop_exn t = pop_or_peek_as t Pop Unit -let size t = +let length t = let tail = Atomic.get t.tail in let head = Atomic.fenceless_get t.head in tail - head diff --git a/test/spsc_queue/dune b/test/spsc_queue/dune index 0483183b..dc8f2d23 100644 --- a/test/spsc_queue/dune +++ b/test/spsc_queue/dune @@ -10,7 +10,7 @@ (rule (action - (copy ../../src/spsc_queue/spsc_queue_intf.ml spsc_queue_intf.ml)) + (copy ../../src/spsc_queue/spsc_queue_intf.mli spsc_queue_intf.ml)) (package saturn)) (test diff --git a/test/spsc_queue/spsc_queue_dscheck.ml b/test/spsc_queue/spsc_queue_dscheck.ml index 10495093..40538f87 100644 --- a/test/spsc_queue/spsc_queue_dscheck.ml +++ b/test/spsc_queue/spsc_queue_dscheck.ml @@ -33,11 +33,11 @@ module Dscheck_spsc (Spsc_queue : Spsc_queue_intf.SPSC_queue) = struct (* ending assertions *) Atomic.final (fun () -> Atomic.check (fun () -> - Spsc_queue.size queue == items_count - !dequeued)) + Spsc_queue.length queue == items_count - !dequeued)) let with_trace ?(shift_by = 0) f () = Atomic.trace (fun () -> f ~shift_by ()) - let size_linearizes_with_1_thr () = + let length_linearizes_with_1_thr () = Atomic.trace (fun () -> let queue = Spsc_queue.create ~size_exponent:4 in Spsc_queue.push_exn queue (-1); @@ -51,11 +51,72 @@ module Dscheck_spsc (Spsc_queue : Spsc_queue_intf.SPSC_queue) = struct let size = ref 0 in Atomic.spawn (fun () -> assert (Option.is_some (Spsc_queue.pop_opt queue)); - size := Spsc_queue.size queue); + size := Spsc_queue.length queue); Atomic.final (fun () -> Atomic.check (fun () -> 1 <= !size && !size <= 5))) + let of_list_exn () = + Atomic.trace (fun () -> + let queue = Spsc_queue.of_list_exn ~size_exponent:4 [ 1; 2; 3 ] in + + Atomic.spawn (fun () -> + for i = 4 to 6 do + Spsc_queue.try_push queue i |> ignore + done); + + let popped1 = ref [] in + let popped2 = ref [] in + Atomic.spawn (fun () -> + for _ = 1 to 3 do + popped1 := Spsc_queue.pop_opt queue :: !popped1 + done; + for _ = 4 to 6 do + popped2 := Spsc_queue.pop_opt queue :: !popped2 + done); + + Atomic.final (fun () -> + Atomic.check (fun () -> + List.map Option.get !popped1 = List.rev [ 1; 2; 3 ]); + Atomic.check (fun () -> + let popped2 = + List.filter Option.is_some !popped2 |> List.map Option.get + in + match popped2 with + | [] | [ 4 ] | [ 5; 4 ] | [ 6; 5; 4 ] -> true + | _ -> false))) + + let drop_exn () = + Atomic.trace (fun () -> + let queue = Spsc_queue.of_list_exn ~size_exponent:4 [ 1; 2; 3 ] in + + Atomic.spawn (fun () -> + for i = 4 to 6 do + Spsc_queue.try_push queue i |> ignore + done); + + let popped1 = ref [] in + let popped2 = ref [] in + Atomic.spawn (fun () -> + Spsc_queue.drop_exn queue; + for _ = 2 to 3 do + popped1 := Spsc_queue.pop_opt queue :: !popped1 + done; + for _ = 4 to 6 do + popped2 := Spsc_queue.pop_opt queue :: !popped2 + done); + + Atomic.final (fun () -> + Atomic.check (fun () -> + List.map Option.get !popped1 = List.rev [ 2; 3 ]); + Atomic.check (fun () -> + let popped2 = + List.filter Option.is_some !popped2 |> List.map Option.get + in + match popped2 with + | [] | [ 4 ] | [ 5; 4 ] | [ 6; 5; 4 ] -> true + | _ -> false))) + let tests name = let open Alcotest in [ @@ -69,9 +130,11 @@ module Dscheck_spsc (Spsc_queue : Spsc_queue_intf.SPSC_queue) = struct (with_trace ~shift_by:s create_test) in [ with_shift 1; with_shift 6; with_shift 11 ] ); - ( "size_" ^ name, - [ test_case "linearizes-with-1-thr" `Slow size_linearizes_with_1_thr ] + ( "length_" ^ name, + [ test_case "linearizes-with-1-thr" `Slow length_linearizes_with_1_thr ] ); + ("of_list_exn_" ^ name, [ test_case "of_list" `Slow of_list_exn ]); + ("drop_exn " ^ name, [ test_case "drop_exn" `Slow drop_exn ]); ] end diff --git a/test/spsc_queue/spsc_queues/dune b/test/spsc_queue/spsc_queues/dune index 2ff250dc..402d2081 100644 --- a/test/spsc_queue/spsc_queues/dune +++ b/test/spsc_queue/spsc_queues/dune @@ -1,6 +1,6 @@ (rule (action - (copy ../../../src/spsc_queue/spsc_queue_intf.ml spsc_queue_intf.ml)) + (copy ../../../src/spsc_queue/spsc_queue_intf.mli spsc_queue_intf.ml)) (package saturn)) (library diff --git a/test/spsc_queue/stm_spsc_queue.ml b/test/spsc_queue/stm_spsc_queue.ml index 9cdc9bee..10c78467 100644 --- a/test/spsc_queue/stm_spsc_queue.ml +++ b/test/spsc_queue/stm_spsc_queue.ml @@ -6,13 +6,14 @@ open Util module STM_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct module Spec = struct - type cmd = Push of int | Pop | Peek + type cmd = Push of int | Pop | Peek | Drop let show_cmd c = match c with | Push i -> "Push " ^ string_of_int i | Pop -> "Pop" | Peek -> "Peek" + | Drop -> "Drop" type state = int * int list type sut = int Spsc_queue.t @@ -23,14 +24,17 @@ module STM_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct let consumer_cmd _s = QCheck.make ~print:show_cmd - (Gen.oneof [ Gen.return Pop; Gen.return Peek ]) + (Gen.oneof [ Gen.return Pop; Gen.return Peek; Gen.return Drop ]) let arb_cmd _s = let int_gen = Gen.nat in QCheck.make ~print:show_cmd (Gen.oneof [ - Gen.return Pop; Gen.return Peek; Gen.map (fun i -> Push i) int_gen; + Gen.return Pop; + Gen.return Peek; + Gen.return Drop; + Gen.map (fun i -> Push i) int_gen; ]) let size_exponent = 4 @@ -42,7 +46,7 @@ module STM_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct let next_state c (n, s) = match c with | Push i -> if n = max_size then (n, s) else (n + 1, i :: s) - | Pop -> ( + | Pop | Drop -> ( match List.rev s with [] -> (0, s) | _ :: s' -> (n - 1, List.rev s')) | Peek -> (n, s) @@ -54,6 +58,7 @@ module STM_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct Res (result unit exn, protect (fun d -> Spsc_queue.push_exn d i) d) | Pop -> Res (result int exn, protect Spsc_queue.pop_exn d) | Peek -> Res (result int exn, protect Spsc_queue.peek_exn d) + | Drop -> Res (result unit exn, protect Spsc_queue.drop_exn d) let postcond c ((n, s) : state) res = match (c, res) with @@ -67,6 +72,11 @@ module STM_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct | Error Spsc_queue.Empty, [] -> true | Ok popped, x :: _ -> x = popped | _ -> false) + | Drop, Res ((Result (Unit, Exn), _), res) -> ( + match (res, List.rev s) with + | Error Spsc_queue.Empty, [] -> true + | Ok (), _ :: _ -> true + | _ -> false) | _, _ -> false end diff --git a/test/spsc_queue/test_spsc_queue.ml b/test/spsc_queue/test_spsc_queue.ml index 19614f3a..6cbc4dc3 100644 --- a/test/spsc_queue/test_spsc_queue.ml +++ b/test/spsc_queue/test_spsc_queue.ml @@ -4,7 +4,7 @@ module Tests_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct let test_empty () = let q = Spsc_queue.create ~size_exponent:3 in assert (Option.is_none (Spsc_queue.pop_opt q)); - assert (Spsc_queue.size q == 0); + assert (Spsc_queue.length q == 0); Printf.printf "test_%s_empty: ok\n" Spsc_queue.name let push_not_full q elt = @@ -18,7 +18,7 @@ module Tests_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct while push_not_full q () do Domain.cpu_relax () done; - assert (Spsc_queue.size q == 8); + assert (Spsc_queue.length q == 8); Printf.printf "test_%s_full: ok\n" Spsc_queue.name let test_parallel () = @@ -48,7 +48,7 @@ module Tests_spsc (Spsc_queue : Spsc_queues.SPSC_tests) = struct last_num := Float.to_int v done; assert (Option.is_none (Spsc_queue.pop_opt q)); - assert (Spsc_queue.size q == 0); + assert (Spsc_queue.length q == 0); Domain.join producer; Printf.printf "test_%s_parallel: ok (transferred = %d)\n" Spsc_queue.name !last_num