(************************************************************************) (* v * The Coq Proof Assistant / The Coq Development Team *) (* t -> bool end module type S = sig type elt (* [may_add_and_get key constr] uses [key] to look for [constr] in the hash table [H]. If [constr] is in [H], returns the specific representation that is stored in [H]. Otherwise, [constr] is stored in [H] and will be used as the canonical representation of this value in the future. *) val may_add_and_get : int -> elt -> elt end module Make (E : Hashtype) = struct type elt = E.t type bucketlist = Empty | Cons of elt * int * bucketlist let initial_size = 19991 let table_data = ref (Array.make initial_size Empty) let table_size = ref 0 let resize () = let odata = !table_data in let osize = Array.length odata in let nsize = min (2 * osize + 1) Sys.max_array_length in if nsize <> osize then begin let ndata = Array.create nsize Empty in let rec insert_bucket = function | Empty -> () | Cons (key, hash, rest) -> let nidx = hash mod nsize in ndata.(nidx) <- Cons (key, hash, ndata.(nidx)); insert_bucket rest in for i = 0 to osize - 1 do insert_bucket odata.(i) done; table_data := ndata end let add hash key = let odata = !table_data in let osize = Array.length odata in let i = hash mod osize in odata.(i) <- Cons (key, hash, odata.(i)); incr table_size; if !table_size > osize lsl 1 then resize () let find_rec hash key bucket = let rec aux = function | Empty -> add hash key; key | Cons (k, h, rest) -> if hash == h && E.equals key k then k else aux rest in aux bucket let may_add_and_get hash key = let odata = !table_data in match odata.(hash mod (Array.length odata)) with | Empty -> add hash key; key | Cons (k1, h1, rest1) -> if hash == h1 && E.equals key k1 then k1 else match rest1 with | Empty -> add hash key; key | Cons (k2, h2, rest2) -> if hash == h2 && E.equals key k2 then k2 else match rest2 with | Empty -> add hash key; key | Cons (k3, h3, rest3) -> if hash == h2 && E.equals key k3 then k3 else find_rec hash key rest3 end module Combine = struct (* These are helper functions to combine the hash keys in a similar way as [Hashtbl.hash] does. The constants [alpha] and [beta] must be prime numbers. There were chosen empirically. Notice that the problem of hashing trees is hard and there are plenty of study on this topic. Therefore, there must be room for improvement here. *) let alpha = 65599 let beta = 7 let combine x y = x * alpha + y let combine3 x y z = combine x (combine y z) let combine4 x y z t = combine x (combine3 y z t) let combinesmall x y = beta * x + y end