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

222 lines
8.2 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 error +=
| Consume_roll_change
| No_roll_in_contract
| Deleted_contract_owning_rolls
2018-02-22 11:28:54 +01:00
| No_roll_snapshot_for_cycle of Cycle_repr.t
2016-09-08 19:13:10 +02:00
let get_contract_delegate c contract =
match Contract_repr.is_implicit contract with
2016-09-08 19:13:10 +02:00
| Some manager -> return (Some manager)
| None -> Storage.Contract.Delegate.get_option c contract
2018-02-22 01:29:40 +01:00
let get_contract_delegate_at_cycle c cycle contract =
match Contract_repr.is_implicit contract with
2018-02-22 01:29:40 +01:00
| Some manager -> return (Some manager)
| None -> Storage.Contract.Delegate.Snapshot.get_option c (cycle, contract)
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.Contract.Delegate.delete_snapshot c cycle >>= fun c ->
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)
| Some contract ->
loop ctxt (Roll_repr.succ roll) (f roll contract acc) in
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.Contract.Delegate.snapshot ctxt cycle >>=? fun ctxt ->
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
| Some contract ->
get_contract_delegate_at_cycle c cycle contract >>=? function
| None ->
loop sequence
| Some delegate ->
Public_key_storage.get_option c delegate >>=? function
| None -> loop sequence
| Some delegate -> return delegate
2018-02-22 01:29:40 +01:00
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
module Contract = struct
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)
let consume_roll_change c contract =
let roll_value = Raw_context.roll_value c in
2016-09-08 19:13:10 +02:00
Storage.Roll.Contract_change.get c contract >>=? fun change ->
trace Consume_roll_change
(Lwt.return Tez_repr.(change -? roll_value)) >>=? fun new_change ->
Storage.Roll.Contract_change.set c contract new_change
2016-09-08 19:13:10 +02:00
let recover_roll_change c contract =
let roll_value = Raw_context.roll_value c in
2016-09-08 19:13:10 +02:00
Storage.Roll.Contract_change.get c contract >>=? fun change ->
Lwt.return Tez_repr.(change +? roll_value) >>=? fun new_change ->
2016-09-08 19:13:10 +02:00
Storage.Roll.Contract_change.set c contract new_change
let pop_roll_from_contract c contract =
recover_roll_change c contract >>=? fun c ->
(* beginning:
contract : roll -> successor_roll -> ...
limbo : limbo_head -> ...
*)
Storage.Roll.Limbo.get_option c >>=? fun limbo_head ->
Storage.Roll.Contract_roll_list.get_option c contract >>=? function
2016-09-08 19:13:10 +02:00
| None -> fail No_roll_in_contract
| Some roll ->
Storage.Roll.Owner.delete c roll >>=? fun c ->
Storage.Roll.Successor.get_option c roll >>=? fun successor_roll ->
Storage.Roll.Contract_roll_list.set_option c contract successor_roll >>= fun c ->
2016-09-08 19:13:10 +02:00
(* contract : successor_roll -> ...
roll ------^
limbo : limbo_head -> ... *)
Storage.Roll.Successor.set_option c roll limbo_head >>= fun c ->
2016-09-08 19:13:10 +02:00
(* contract : successor_roll -> ...
roll ------v
limbo : limbo_head -> ... *)
Storage.Roll.Limbo.init_set c roll >>= fun c ->
2016-09-08 19:13:10 +02:00
(* contract : successor_roll -> ...
limbo : roll -> limbo_head -> ... *)
return (roll, c)
2016-09-08 19:13:10 +02:00
let create_roll_in_contract c contract =
consume_roll_change c contract >>=? fun c ->
(* beginning:
contract : contract_head -> ...
limbo : roll -> limbo_successor -> ...
*)
Storage.Roll.Contract_roll_list.get_option c contract >>=? fun contract_head ->
2016-09-08 19:13:10 +02:00
get_limbo_roll c >>=? fun (roll, c) ->
Storage.Roll.Owner.init c roll contract >>=? fun c ->
Storage.Roll.Successor.get_option c roll >>=? fun limbo_successor ->
Storage.Roll.Limbo.set_option c limbo_successor >>= fun c ->
2016-09-08 19:13:10 +02:00
(* contract : contract_head -> ...
roll ------v
limbo : limbo_successor -> ... *)
Storage.Roll.Successor.set_option c roll contract_head >>= fun c ->
2016-09-08 19:13:10 +02:00
(* contract : contract_head -> ...
roll ------^
limbo : limbo_successor -> ... *)
Storage.Roll.Contract_roll_list.init_set c contract roll >>= fun c ->
(* contract : roll -> contract_head -> ...
limbo : limbo_successor -> ... *)
return c
2016-09-08 19:13:10 +02:00
let init c contract =
Storage.Roll.Contract_change.init c contract Tez_repr.zero
let add_amount c contract amount =
let roll_value = Raw_context.roll_value c in
2016-09-08 19:13:10 +02:00
Storage.Roll.Contract_change.get c contract >>=? fun change ->
Lwt.return Tez_repr.(amount +? change) >>=? fun change ->
Storage.Roll.Contract_change.set c contract change >>=? fun c ->
let rec loop c change =
if Tez_repr.(change < roll_value) then
return c
else
Lwt.return Tez_repr.(change -? roll_value) >>=? fun change ->
create_roll_in_contract c contract >>=? fun c ->
loop c change in
2016-09-08 19:13:10 +02:00
loop c change
let remove_amount c contract 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
pop_roll_from_contract c contract >>=? fun (_, c) ->
Lwt.return Tez_repr.(change +? roll_value) >>=? fun change ->
loop c change in
2016-09-08 19:13:10 +02:00
Storage.Roll.Contract_change.get c contract >>=? fun change ->
loop c change >>=? fun (c, change) ->
Lwt.return Tez_repr.(change -? amount) >>=? fun change ->
2017-06-30 01:58:47 +02:00
Storage.Roll.Contract_roll_list.mem c contract >>= fun rolls ->
if Tez_repr.(change = zero) && not rolls then
2017-05-09 17:35:56 +02:00
Storage.Roll.Contract_change.delete c contract
else
Storage.Roll.Contract_change.set c contract change
2016-09-08 19:13:10 +02:00
let assert_empty c contract =
2017-05-09 17:35:56 +02:00
Storage.Roll.Contract_change.mem c contract >>= fun change ->
2017-06-30 01:58:47 +02:00
fail_unless (not change) Deleted_contract_owning_rolls
2016-09-08 19:13:10 +02:00
end
let value = Raw_context.roll_value
2016-09-08 19:13:10 +02:00
let init c =
Storage.Roll.Next.init c Roll_repr.first