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 =
|
2018-02-21 18:08:09 +01:00
|
|
|
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 =
|
2018-02-21 18:08:09 +01:00
|
|
|
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 ->
|
2017-11-19 18:00:04 +01:00
|
|
|
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
|
2017-04-10 12:50:15 +02:00
|
|
|
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 ->
|
2018-02-22 13:53:07 -05:00
|
|
|
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 =
|
2017-04-10 12:50:15 +02:00
|
|
|
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 =
|
2017-11-16 16:45:22 +01:00
|
|
|
Storage.Roll.Limbo.get_option c >>=? function
|
2016-09-08 19:13:10 +02:00
|
|
|
| None ->
|
|
|
|
fresh_roll c >>=? fun (roll, c) ->
|
2017-11-16 16:45:22 +01:00
|
|
|
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 =
|
2017-11-19 21:32:24 +01:00
|
|
|
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 ->
|
2017-03-10 14:39:22 +01:00
|
|
|
trace Consume_roll_change
|
2017-11-19 21:32:24 +01:00
|
|
|
(Lwt.return Tez_repr.(change -? roll_value)) >>=? fun new_change ->
|
2017-03-10 14:39:22 +01:00
|
|
|
Storage.Roll.Contract_change.set c contract new_change
|
2016-09-08 19:13:10 +02:00
|
|
|
|
|
|
|
let recover_roll_change c contract =
|
2017-11-19 21:32:24 +01:00
|
|
|
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 ->
|
2017-11-19 21:32:24 +01:00
|
|
|
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 -> ...
|
|
|
|
*)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 ->
|
2017-11-16 16:45:22 +01:00
|
|
|
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 -> ... *)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 -> ... *)
|
2017-11-16 16:45:22 +01:00
|
|
|
Storage.Roll.Limbo.init_set c roll >>= fun c ->
|
2016-09-08 19:13:10 +02:00
|
|
|
(* contract : successor_roll -> ...
|
|
|
|
limbo : roll -> limbo_head -> ... *)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 -> ...
|
|
|
|
*)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 ->
|
2017-11-16 16:45:22 +01:00
|
|
|
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 -> ... *)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 -> ... *)
|
2017-11-16 16:45:22 +01:00
|
|
|
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 =
|
2017-11-19 21:32:24 +01:00
|
|
|
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 =
|
2017-11-19 21:32:24 +01:00
|
|
|
if Tez_repr.(change < roll_value) then
|
2017-03-10 14:39:22 +01:00
|
|
|
return c
|
|
|
|
else
|
2017-11-19 21:32:24 +01:00
|
|
|
Lwt.return Tez_repr.(change -? roll_value) >>=? fun change ->
|
2017-03-10 14:39:22 +01:00
|
|
|
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 =
|
2017-11-19 21:32:24 +01:00
|
|
|
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)
|
2017-03-10 14:39:22 +01:00
|
|
|
then return (c, change)
|
2016-09-08 19:13:10 +02:00
|
|
|
else
|
|
|
|
pop_roll_from_contract c contract >>=? fun (_, c) ->
|
2017-11-19 21:32:24 +01:00
|
|
|
Lwt.return Tez_repr.(change +? roll_value) >>=? fun change ->
|
2017-03-10 14:39:22 +01:00
|
|
|
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) ->
|
2017-03-10 14:39:22 +01:00
|
|
|
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
|
|
|
|
|
2017-11-19 21:32:24 +01:00
|
|
|
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
|