ligo/src/lib_crypto/blake2B.ml

284 lines
7.3 KiB
OCaml
Raw Normal View History

2016-09-08 21:13:10 +04:00
(**************************************************************************)
(* *)
2018-02-06 00:17:03 +04:00
(* Copyright (c) 2014 - 2018. *)
2016-09-08 21:13:10 +04:00
(* Dynamic Ledger Solutions, Inc. <contact@tezos.com> *)
(* *)
(* All rights reserved. No warranty, explicit or implicit, provided. *)
(* *)
(**************************************************************************)
(*-- Type specific Hash builder ---------------------------------------------*)
module type Name = sig
val name : string
val title : string
val size : int option
end
module type PrefixedName = sig
include Name
val b58check_prefix : string
end
module Make_minimal (K : Name) = struct
open Blake2
type t = Blake2b.hash
2016-09-08 21:13:10 +04:00
include K
let size =
match K.size with
| None -> 32
| Some x -> x
2016-09-08 21:13:10 +04:00
let of_string s =
if String.length s <> size then
None
else
Some (Blake2b.Hash (Cstruct.of_string s))
let of_string_exn s =
match of_string s with
| None ->
let msg =
Printf.sprintf "%s.of_string: wrong string size (%d)"
K.name (String.length s) in
raise (Invalid_argument msg)
| Some h -> h
let to_string (Blake2b.Hash h) = Cstruct.to_string h
2016-09-08 21:13:10 +04:00
let of_hex s = of_string (Hex.to_string (`Hex s))
let of_hex_exn s = of_string_exn (Hex.to_string (`Hex s))
let to_hex s =
let `Hex s = Hex.of_string (to_string s) in
s
2016-09-08 21:13:10 +04:00
let compare (Blake2b.Hash h1) (Blake2b.Hash h2) = Cstruct.compare h1 h2
let equal x y = compare x y = 0
2016-09-08 21:13:10 +04:00
let of_bytes_opt b =
if MBytes.length b <> size then
None
else
Some (Blake2b.Hash (Cstruct.of_bigarray b))
let of_bytes_exn b =
match of_bytes_opt b with
| None ->
let msg =
Printf.sprintf "%s.of_bytes: wrong string size (%d)"
K.name (MBytes.length b) in
raise (Invalid_argument msg)
| Some h -> h
let to_bytes (Blake2b.Hash h) = Cstruct.to_bigarray h
2016-09-08 21:13:10 +04:00
let read src off = of_bytes_exn @@ MBytes.sub src off size
let write dst off h = MBytes.blit (to_bytes h) 0 dst off size
2016-09-08 21:13:10 +04:00
let hash_bytes l =
let state = Blake2b.init size in
List.iter (fun b -> Blake2b.update state (Cstruct.of_bigarray b)) l ;
Blake2b.final state
2016-09-08 21:13:10 +04:00
let hash_string l =
let state = Blake2b.init size in
List.iter (fun s -> Blake2b.update state (Cstruct.of_string s)) l ;
Blake2b.final state
2016-09-08 21:13:10 +04:00
let path_length = 6
let to_path key l =
2016-09-08 21:13:10 +04:00
let key = to_hex key in
String.sub key 0 2 :: String.sub key 2 2 ::
String.sub key 4 2 :: String.sub key 6 2 ::
String.sub key 8 2 :: String.sub key 10 (size * 2 - 10) :: l
2016-09-08 21:13:10 +04:00
let of_path path =
let path = String.concat "" path in
of_hex path
let of_path_exn path =
let path = String.concat "" path in
of_hex_exn path
2016-09-08 21:13:10 +04:00
2016-10-06 20:30:04 +04:00
let prefix_path p =
let `Hex p = Hex.of_string p in
2016-10-06 20:30:04 +04:00
let len = String.length p in
let p1 = if len >= 2 then String.sub p 0 2 else ""
and p2 = if len >= 4 then String.sub p 2 2 else ""
and p3 = if len >= 6 then String.sub p 4 2 else ""
and p4 = if len >= 8 then String.sub p 6 2 else ""
and p5 = if len >= 10 then String.sub p 8 2 else ""
and p6 = if len > 10 then String.sub p 10 (len - 10) else "" in
[ p1 ; p2 ; p3 ; p4 ; p5 ; p6 ]
end
module Make (R : sig
val register_encoding:
prefix: string ->
length:int ->
to_raw: ('a -> string) ->
of_raw: (string -> 'a option) ->
wrap: ('a -> Base58.data) ->
'a Base58.encoding
end) (K : PrefixedName) = struct
include Make_minimal(K)
2016-10-06 20:30:04 +04:00
let zero =
match of_hex (String.make (size * 2) '0') with
| Some c -> c
| None -> assert false
2016-09-08 21:13:10 +04:00
(* Serializers *)
type Base58.data += Hash of t
let b58check_encoding =
R.register_encoding
~prefix: K.b58check_prefix
~length:size
~wrap: (fun s -> Hash s)
~of_raw:(fun h -> of_string h) ~to_raw:to_string
let of_b58check_opt s =
Base58.simple_decode b58check_encoding s
let of_b58check_exn s =
match Base58.simple_decode b58check_encoding s with
| Some x -> x
| None -> Format.kasprintf Pervasives.failwith "Unexpected hash (%s)" K.name
let to_b58check s = Base58.simple_encode b58check_encoding s
let to_short_b58check s =
String.sub (to_b58check s) 0 (10 + 2 * String.length K.b58check_prefix)
2016-09-08 21:13:10 +04:00
let pp ppf t =
Format.pp_print_string ppf (to_b58check t)
2016-09-08 21:13:10 +04:00
let pp_short ppf t =
Format.pp_print_string ppf (to_short_b58check t)
2016-09-08 21:13:10 +04:00
end
2017-03-16 20:17:06 +04:00
module Generic_Merkle_tree (H : sig
type t
type elt
val empty : t
val leaf : elt -> t
val node : t -> t -> t
end) = struct
let rec step a n =
let m = (n+1) / 2 in
for i = 0 to m - 1 do
a.(i) <- H.node a.(2*i) a.(2*i+1)
done ;
a.(m) <- H.node a.(n) a.(n) ;
if m = 1 then
a.(0)
else if m mod 2 = 0 then
step a m
else begin
a.(m+1) <- a.(m) ;
step a (m+1)
end
let empty = H.empty
let compute xs =
match xs with
| [] -> H.empty
| [x] -> H.leaf x
| _ :: _ :: _ ->
let last = TzList.last_exn xs in
2017-03-16 20:17:06 +04:00
let n = List.length xs in
let a = Array.make (n+1) (H.leaf last) in
List.iteri (fun i x -> a.(i) <- H.leaf x) xs ;
step a n
type path =
| Left of path * H.t
| Right of H.t * path
| Op
let rec step_path a n p j =
let m = (n+1) / 2 in
let p = if j mod 2 = 0 then Left (p, a.(j+1)) else Right (a.(j-1), p) in
for i = 0 to m - 1 do
a.(i) <- H.node a.(2*i) a.(2*i+1)
done ;
a.(m) <- H.node a.(n) a.(n) ;
if m = 1 then
p
else if m mod 2 = 0 then
step_path a m p (j/2)
else begin
a.(m+1) <- a.(m) ;
step_path a (m+1) p (j/2)
end
let compute_path xs i =
match xs with
| [] -> invalid_arg "compute_path"
| [_] -> Op
| _ :: _ :: _ ->
let last = TzList.last_exn xs in
2017-03-16 20:17:06 +04:00
let n = List.length xs in
if i < 0 || n <= i then invalid_arg "compute_path" ;
let a = Array.make (n+1) (H.leaf last) in
List.iteri (fun i x -> a.(i) <- H.leaf x) xs ;
step_path a n Op i
let rec check_path p h =
match p with
| Op ->
H.leaf h, 1, 0
| Left (p, r) ->
let l, s, pos = check_path p h in
H.node l r, s * 2, pos
| Right (l, p) ->
let r, s, pos = check_path p h in
H.node l r, s * 2, pos + s
let check_path p h =
let h, _, pos = check_path p h in
h, pos
2016-09-08 21:13:10 +04:00
end
2017-03-16 20:17:06 +04:00
module Make_merkle_tree
(R : sig
val register_encoding:
prefix: string ->
length:int ->
to_raw: ('a -> string) ->
of_raw: (string -> 'a option) ->
wrap: ('a -> Base58.data) ->
'a Base58.encoding
end)
(K : PrefixedName)
2017-03-16 20:17:06 +04:00
(Contents: sig
type t
val to_bytes: t -> MBytes.t
end) = struct
include Make (R) (K)
2017-03-16 20:17:06 +04:00
type elt = Contents.t
let elt_bytes = Contents.to_bytes
2017-03-16 20:17:06 +04:00
let empty = hash_bytes []
include Generic_Merkle_tree(struct
type nonrec t = t
type nonrec elt = elt
let empty = empty
let leaf x = hash_bytes [Contents.to_bytes x]
let node x y = hash_bytes [to_bytes x; to_bytes y]
end)
2016-09-08 21:13:10 +04:00
2017-03-16 20:17:06 +04:00
end
2016-09-08 21:13:10 +04:00
include
Make_minimal (struct
let name = "Generic_hash"
let title = ""
let size = None
end)