98 lines
3.1 KiB
OCaml
Raw Normal View History

2016-09-08 19:13:10 +02:00
(**************************************************************************)
(* *)
2018-02-05 21:17:03 +01:00
(* Copyright (c) 2014 - 2018. *)
2016-09-08 19:13:10 +02:00
(* Dynamic Ledger Solutions, Inc. <contact@tezos.com> *)
(* *)
(* All rights reserved. No warranty, explicit or implicit, provided. *)
(* *)
(**************************************************************************)
type location = Micheline.canonical_location
2016-09-08 19:13:10 +02:00
let location_encoding = Micheline.canonical_location_encoding
2016-09-08 19:13:10 +02:00
type expr = Michelson_v1_primitives.prim Micheline.canonical
2016-09-08 19:13:10 +02:00
2018-05-04 15:05:20 +02:00
type lazy_expr = expr Data_encoding.lazy_t
type node = (location, Michelson_v1_primitives.prim) Micheline.node
2016-09-08 19:13:10 +02:00
2018-03-14 11:36:09 +01:00
let expr_encoding =
Micheline.canonical_encoding
~variant:"michelson_v1"
Michelson_v1_primitives.prim_encoding
2016-09-08 19:13:10 +02:00
2018-05-04 15:05:20 +02:00
type error += Lazy_script_decode (* `Permanent *)
let () =
register_error_kind `Permanent
~id:"invalid_binary_format"
~title:"Invalid binary format"
~description:"Could not deserialize some piece of data \
from its binary representation"
Data_encoding.empty
(function Lazy_script_decode -> Some () | _ -> None)
(fun () -> Lazy_script_decode)
let lazy_expr_encoding =
Data_encoding.lazy_encoding expr_encoding
let lazy_expr expr =
Data_encoding.make_lazy expr_encoding expr
let force_decode expr =
match Data_encoding.force_decode expr with
| Some v -> ok v
| None -> error Lazy_script_decode
let force_bytes expr =
match Data_encoding.force_bytes expr with
| bytes -> ok bytes
| exception _ -> error Lazy_script_decode
type t = {
code : lazy_expr ;
storage : lazy_expr
}
2016-09-08 19:13:10 +02:00
let encoding =
let open Data_encoding in
def "scripted.contracts" @@
conv
(fun { code ; storage } -> (code, storage))
(fun (code, storage) -> { code ; storage })
2018-05-04 15:05:20 +02:00
(obj2
(req "code" lazy_expr_encoding)
(req "storage" lazy_expr_encoding))
let rec node_size node =
let open Micheline in
match node with
| Int (_, n) -> (1, 1 + (Z.numbits n + 63) / 64)
| String (_, s) -> (1, 1 + (String.length s + 7) / 8)
| Prim (_, _, args, annot) ->
List.fold_left
(fun (blocks, words) node ->
let (nblocks, nwords) = node_size node in
(blocks + 1 + nblocks, words + 2 + nwords))
(match annot with
2018-05-17 19:37:25 +02:00
| [] -> (1, 2)
| annots ->
let annots_length = List.fold_left (fun acc s -> acc + String.length s) 0 annots in
(1, 4 + (annots_length + 7) / 8))
args
2018-05-17 19:37:25 +02:00
| Seq (_, args) ->
List.fold_left
(fun (blocks, words) node ->
let (nblocks, nwords) = node_size node in
(blocks + 1 + nblocks, words + 2 + nwords))
2018-05-17 19:37:25 +02:00
(1, 2)
args
let expr_size expr =
node_size (Micheline.root expr)
let expr_cost expr =
let blocks, words = expr_size expr in
Gas_limit_repr.(((Compare.Int.max 0 (blocks - 1)) *@ alloc_cost 0) +@ alloc_cost words)