ligo/src/proto_alpha/lib_protocol/src/roll_storage.ml

268 lines
9.6 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. *)
(* *)
(**************************************************************************)
open Misc
2016-09-08 19:13:10 +02:00
type error +=
| Consume_roll_change
2018-02-21 22:52:33 +01:00
| No_roll_for_delegate
2018-02-22 11:28:54 +01:00
| No_roll_snapshot_for_cycle of Cycle_repr.t
2018-03-13 11:13:12 +01:00
| Unregistered_delegate of Ed25519.Public_key_hash.t (* `Permanent *)
let () =
register_error_kind
`Permanent
2018-03-13 11:13:12 +01:00
~id:"contract.manager.unregistered_delegate"
~title:"Unregistered delegate"
~description:"A contract cannot be delegated to an unregistered delegate"
~pp:(fun ppf (k) ->
Format.fprintf ppf "The provided public key (with hash %a) is \
2018-03-13 11:13:12 +01:00
\ not registered as valid delegate key."
Ed25519.Public_key_hash.pp k)
Data_encoding.(obj1 (req "hash" Ed25519.Public_key_hash.encoding))
2018-03-13 11:13:12 +01:00
(function Unregistered_delegate (k) -> Some (k) | _ -> None)
(fun (k) -> Unregistered_delegate (k))
2016-09-08 19:13:10 +02:00
let get_contract_delegate c contract =
Storage.Contract.Delegate.get_option c contract
2016-09-08 19:13:10 +02:00
let delegate_pubkey ctxt delegate =
Storage.Contract.Manager.get_option ctxt
(Contract_repr.implicit_contract delegate) >>=? function
| None | Some (Manager_repr.Hash _) ->
2018-03-13 11:13:12 +01:00
fail (Unregistered_delegate delegate)
| Some (Manager_repr.Public_key pk) ->
return pk
2018-02-22 01:29:40 +01:00
2016-09-08 19:13:10 +02:00
let clear_cycle c cycle =
Storage.Roll.Last_for_cycle.delete c cycle >>=? fun c ->
2018-02-22 01:29:40 +01:00
Storage.Roll.Owner.delete_snapshot c cycle >>= fun c ->
return c
2016-09-08 19:13:10 +02:00
let fold ctxt ~f init =
Storage.Roll.Next.get ctxt >>=? fun last ->
let rec loop ctxt roll acc =
acc >>=? fun acc ->
if Roll_repr.(roll = last) then
return acc
else
Storage.Roll.Owner.get_option ctxt roll >>=? function
| None ->
loop ctxt (Roll_repr.succ roll) (return acc)
2018-02-21 22:52:33 +01:00
| Some delegate ->
loop ctxt (Roll_repr.succ roll) (f roll delegate acc) in
2016-09-08 19:13:10 +02:00
loop ctxt Roll_repr.first (return init)
2016-10-26 17:02:10 +02:00
let freeze_rolls_for_cycle ctxt cycle =
2018-02-22 01:29:40 +01:00
Storage.Roll.Owner.snapshot ctxt cycle >>=? fun ctxt ->
Storage.Roll.Next.get ctxt >>=? fun last ->
Storage.Roll.Last_for_cycle.init ctxt cycle last
2016-09-08 19:13:10 +02:00
(* Roll selection *)
module Random = struct
let int32_to_bytes i =
let b = MBytes.create 4 in
MBytes.set_int32 b 0 i;
b
let level_random seed use level =
let position = level.Level_repr.cycle_position in
Seed_repr.initialize_new seed
[MBytes.of_string ("level "^use^":");
int32_to_bytes position]
let owner c kind level offset =
let cycle = level.Level_repr.cycle in
Seed_storage.for_cycle c cycle >>=? fun random_seed ->
let rd = level_random random_seed kind level in
let sequence = Seed_repr.sequence rd (Int32.of_int offset) in
2016-09-08 19:13:10 +02:00
Storage.Roll.Last_for_cycle.get c cycle >>=? fun bound ->
2018-02-22 01:29:40 +01:00
let rec loop sequence =
let roll, sequence = Roll_repr.random sequence ~bound in
Storage.Roll.Owner.Snapshot.get_option c (cycle, roll) >>=? function
| None ->
loop sequence
2018-02-21 22:52:33 +01:00
| Some delegate ->
return delegate in
2018-02-22 11:28:54 +01:00
Storage.Roll.Owner.snapshot_exists c cycle >>= fun snapshot_exists ->
fail_unless snapshot_exists (No_roll_snapshot_for_cycle cycle) >>=? fun () ->
2018-02-22 01:29:40 +01:00
loop sequence
2016-09-08 19:13:10 +02:00
end
2017-11-01 04:07:33 -07:00
let baking_rights_owner c level ~priority =
Random.owner c "baking" level priority
2016-09-08 19:13:10 +02:00
let endorsement_rights_owner c level ~slot =
Random.owner c "endorsement" level slot
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
module Delegate = struct
2016-09-08 19:13:10 +02:00
let fresh_roll c =
Storage.Roll.Next.get c >>=? fun roll ->
Storage.Roll.Next.set c (Roll_repr.succ roll) >>=? fun c ->
return (roll, c)
let get_limbo_roll c =
Storage.Roll.Limbo.get_option c >>=? function
2016-09-08 19:13:10 +02:00
| None ->
fresh_roll c >>=? fun (roll, c) ->
Storage.Roll.Limbo.init c roll >>=? fun c ->
2016-09-08 19:13:10 +02:00
return (roll, c)
| Some roll ->
return (roll, c)
2018-02-21 22:52:33 +01:00
let consume_roll_change c delegate =
let roll_value = Raw_context.roll_value c in
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.get c delegate >>=? fun change ->
trace Consume_roll_change
(Lwt.return Tez_repr.(change -? roll_value)) >>=? fun new_change ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.set c delegate new_change
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
let recover_roll_change c delegate =
let roll_value = Raw_context.roll_value c in
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.get c delegate >>=? fun change ->
Lwt.return Tez_repr.(change +? roll_value) >>=? fun new_change ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.set c delegate new_change
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
let pop_roll_from_delegate c delegate =
recover_roll_change c delegate >>=? fun c ->
2016-09-08 19:13:10 +02:00
(* beginning:
2018-02-21 22:52:33 +01:00
delegate : roll -> successor_roll -> ...
2016-09-08 19:13:10 +02:00
limbo : limbo_head -> ...
*)
Storage.Roll.Limbo.get_option c >>=? fun limbo_head ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_roll_list.get_option c delegate >>=? function
| None -> fail No_roll_for_delegate
2016-09-08 19:13:10 +02:00
| Some roll ->
Storage.Roll.Owner.delete c roll >>=? fun c ->
Storage.Roll.Successor.get_option c roll >>=? fun successor_roll ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_roll_list.set_option c delegate successor_roll >>= fun c ->
(* delegate : successor_roll -> ...
2016-09-08 19:13:10 +02:00
roll ------^
limbo : limbo_head -> ... *)
Storage.Roll.Successor.set_option c roll limbo_head >>= fun c ->
2018-02-21 22:52:33 +01:00
(* delegate : successor_roll -> ...
2016-09-08 19:13:10 +02:00
roll ------v
limbo : limbo_head -> ... *)
Storage.Roll.Limbo.init_set c roll >>= fun c ->
2018-02-21 22:52:33 +01:00
(* delegate : successor_roll -> ...
2016-09-08 19:13:10 +02:00
limbo : roll -> limbo_head -> ... *)
return (roll, c)
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
let create_roll_in_delegate c delegate delegate_pk =
consume_roll_change c delegate >>=? fun c ->
2016-09-08 19:13:10 +02:00
(* beginning:
2018-02-21 22:52:33 +01:00
delegate : delegate_head -> ...
2016-09-08 19:13:10 +02:00
limbo : roll -> limbo_successor -> ...
*)
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_roll_list.get_option c delegate >>=? fun delegate_head ->
2016-09-08 19:13:10 +02:00
get_limbo_roll c >>=? fun (roll, c) ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Owner.init c roll delegate_pk >>=? fun c ->
Storage.Roll.Successor.get_option c roll >>=? fun limbo_successor ->
Storage.Roll.Limbo.set_option c limbo_successor >>= fun c ->
2018-02-21 22:52:33 +01:00
(* delegate : delegate_head -> ...
2016-09-08 19:13:10 +02:00
roll ------v
limbo : limbo_successor -> ... *)
2018-02-21 22:52:33 +01:00
Storage.Roll.Successor.set_option c roll delegate_head >>= fun c ->
(* delegate : delegate_head -> ...
2016-09-08 19:13:10 +02:00
roll ------^
limbo : limbo_successor -> ... *)
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_roll_list.init_set c delegate roll >>= fun c ->
(* delegate : roll -> delegate_head -> ...
limbo : limbo_successor -> ... *)
return c
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
let ensure_inited c delegate =
Storage.Roll.Delegate_change.mem c delegate >>= function
| true -> return c
| false ->
Storage.Roll.Delegate_change.init c delegate Tez_repr.zero
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
let add_amount c delegate amount =
ensure_inited c delegate >>=? fun c ->
let roll_value = Raw_context.roll_value c in
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.get c delegate >>=? fun change ->
2016-09-08 19:13:10 +02:00
Lwt.return Tez_repr.(amount +? change) >>=? fun change ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.set c delegate change >>=? fun c ->
delegate_pubkey c delegate >>=? fun delegate_pk ->
2016-09-08 19:13:10 +02:00
let rec loop c change =
if Tez_repr.(change < roll_value) then
return c
else
Lwt.return Tez_repr.(change -? roll_value) >>=? fun change ->
2018-02-21 22:52:33 +01:00
create_roll_in_delegate c delegate delegate_pk >>=? fun c ->
loop c change in
2016-09-08 19:13:10 +02:00
loop c change
2018-02-21 22:52:33 +01:00
let remove_amount c delegate amount =
let roll_value = Raw_context.roll_value c in
2016-09-08 19:13:10 +02:00
let rec loop c change =
if Tez_repr.(amount <= change)
then return (c, change)
2016-09-08 19:13:10 +02:00
else
2018-02-21 22:52:33 +01:00
pop_roll_from_delegate c delegate >>=? fun (_, c) ->
Lwt.return Tez_repr.(change +? roll_value) >>=? fun change ->
loop c change in
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.get c delegate >>=? fun change ->
2016-09-08 19:13:10 +02:00
loop c change >>=? fun (c, change) ->
Lwt.return Tez_repr.(change -? amount) >>=? fun change ->
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_roll_list.mem c delegate >>= fun rolls ->
2017-06-30 01:58:47 +02:00
if Tez_repr.(change = zero) && not rolls then
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.delete c delegate
2017-05-09 17:35:56 +02:00
else
2018-02-21 22:52:33 +01:00
Storage.Roll.Delegate_change.set c delegate change
end
2016-09-08 19:13:10 +02:00
2018-02-21 22:52:33 +01:00
module Contract = struct
let add_amount c contract amount =
get_contract_delegate c contract >>=? function
| None -> return c
| Some delegate ->
Delegate.add_amount c delegate amount
let remove_amount c contract amount =
get_contract_delegate c contract >>=? function
| None -> return c
| Some delegate ->
Delegate.remove_amount c delegate amount
2016-09-08 19:13:10 +02:00
end
let value = Raw_context.roll_value
let init ctxt =
Storage.Roll.Next.init ctxt Roll_repr.first
let init_first_cycles ctxt =
let preserved = Constants_storage.preserved_cycles ctxt in
List.fold_left
(fun ctxt c ->
ctxt >>=? fun ctxt ->
let cycle = Cycle_repr.of_int32_exn (Int32.of_int c) in
freeze_rolls_for_cycle ctxt cycle)
(return ctxt) (0 --> (preserved + 1)) >>=? fun ctxt ->
return ctxt
let cycle_end ctxt last_cycle =
let preserved = Constants_storage.preserved_cycles ctxt in
begin
match Cycle_repr.sub last_cycle preserved with
| None -> return ctxt
| Some cleared_cycle ->
clear_cycle ctxt cleared_cycle
end >>=? fun ctxt ->
let frozen_roll_cycle = Cycle_repr.add last_cycle (preserved+2) in
freeze_rolls_for_cycle ctxt frozen_roll_cycle