(* * Copyright (c) Facebook, Inc. and its affiliates. * * This source code is licensed under the MIT license found in the * LICENSE file in the root directory of this source tree. *) open! NS0 include NSSet_intf type ('elt, 'compare_elt) t = ('elt, 'compare_elt) Set.t [@@deriving compare, equal, sexp] type 'compare_elt compare = 'compare_elt Set.compare [@@deriving compare, equal, sexp] module Make_from_Comparer (Elt : sig type t [@@deriving equal, sexp_of] include Comparer.S with type t := t end) = struct module S = Set.Make [@inlined] (Elt) type elt = Elt.t type compare_elt = Elt.compare type t = S.t [@@deriving compare] type compare = S.compare [@@deriving compare, equal, sexp] let comparer = S.comparer include S.Provide_equal (Elt) module Provide_hash (Elt : sig type t = elt [@@deriving hash] end) = struct let hash_fold_t h s = let length = ref 0 in let s = S.fold (fun x h -> incr length ; Elt.hash_fold_t h x ) s h in Hash.fold_int s !length let hash = Hash.of_fold hash_fold_t end include S.Provide_sexp_of (Elt) module Provide_of_sexp = S.Provide_of_sexp let empty = S.empty let of_ = S.singleton let of_option xo = Option.map_or ~f:S.singleton xo ~default:empty let add x s = S.add x s let add_option = Option.fold ~f:add let add_list xs s = S.union (S.of_list xs) s let remove x s = S.remove x s let diff = S.diff let inter = S.inter let union = S.union let diff_inter s t = (diff s t, inter s t) let diff_inter_diff s t = (diff s t, inter s t, diff t s) let union_list ss = List.fold ~f:union ss empty let is_empty = S.is_empty let cardinal = S.cardinal let mem = S.mem let subset s ~of_:t = S.subset s t let disjoint = S.disjoint let max_elt = S.max_elt_opt let root_elt s = let exception Found in let found = ref None in try S.for_all (fun elt -> found := Some elt ; raise_notrace Found ) s |> ignore ; None with Found -> !found let choose = root_elt let choose_exn m = Option.get_exn (choose m) let only_elt s = match root_elt s with | Some elt -> ( match S.split elt s with | l, _, r when is_empty l && is_empty r -> Some elt | _ -> None ) | None -> None let classify s = match root_elt s with | None -> `Zero | Some elt -> ( match S.split elt s with | l, true, r when is_empty l && is_empty r -> `One elt | _ -> `Many ) let pop s = match choose s with | Some elt -> Some (elt, S.remove elt s) | None -> None let pop_exn s = let elt = choose_exn s in (elt, S.remove elt s) let map s ~f = S.map f s let filter s ~f = S.filter f s let partition s ~f = S.partition f s let iter s ~f = S.iter f s let exists s ~f = S.exists f s let for_all s ~f = S.for_all f s let fold s z ~f = S.fold f s z let reduce xs ~f = match pop xs with Some (x, xs) -> Some (fold ~f xs x) | None -> None let to_list = S.elements let of_list = S.of_list let to_iter s = Iter.from_iter (fun f -> S.iter f s) let of_iter s = Iter.fold ~f:add s S.empty let pp_full ?pre ?suf ?(sep = (",@ " : (unit, unit) fmt)) pp_elt fs x = List.pp ?pre ?suf sep pp_elt fs (S.elements x) module Provide_pp (Elt : sig type t = elt val pp : t pp end) = struct let pp = pp_full Elt.pp let pp_diff fs (xs, ys) = let lose = diff xs ys and gain = diff ys xs in if not (is_empty lose) then Format.fprintf fs "-- %a" pp lose ; if not (is_empty gain) then Format.fprintf fs "++ %a" pp gain end end [@@inline] module Make (Elt : sig type t [@@deriving compare, equal, sexp_of] end) = Make_from_Comparer (struct include Elt include Comparer.Make (Elt) end) [@@inline]