2020-09-19 03:46:03 +00:00
|
|
|
package power
|
|
|
|
|
|
|
|
import (
|
2020-10-28 23:52:52 +00:00
|
|
|
"bytes"
|
|
|
|
|
2020-09-19 03:46:03 +00:00
|
|
|
"github.com/filecoin-project/go-address"
|
|
|
|
"github.com/filecoin-project/go-state-types/abi"
|
2020-09-24 00:40:29 +00:00
|
|
|
"github.com/ipfs/go-cid"
|
2020-10-28 23:52:52 +00:00
|
|
|
cbg "github.com/whyrusleeping/cbor-gen"
|
2020-09-24 00:40:29 +00:00
|
|
|
|
2020-09-23 22:36:56 +00:00
|
|
|
"github.com/filecoin-project/lotus/chain/actors/adt"
|
2020-09-19 03:46:03 +00:00
|
|
|
"github.com/filecoin-project/lotus/chain/actors/builtin"
|
2020-09-23 22:36:56 +00:00
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
power2 "github.com/filecoin-project/specs-actors/v2/actors/builtin/power"
|
|
|
|
adt2 "github.com/filecoin-project/specs-actors/v2/actors/util/adt"
|
2020-09-19 03:46:03 +00:00
|
|
|
)
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
var _ State = (*state2)(nil)
|
2020-09-19 03:46:03 +00:00
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func load2(store adt.Store, root cid.Cid) (State, error) {
|
|
|
|
out := state2{store: store}
|
2020-09-24 00:40:29 +00:00
|
|
|
err := store.Get(store.Context(), root, &out)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &out, nil
|
|
|
|
}
|
|
|
|
|
2021-05-15 01:11:23 +00:00
|
|
|
func make2(store adt.Store) (State, error) {
|
|
|
|
out := state2{store: store}
|
|
|
|
|
2022-04-07 18:46:23 +00:00
|
|
|
em, err := adt2.MakeEmptyMap(store).Root()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
emm, err := adt2.MakeEmptyMultimap(store).Root()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2021-05-15 01:11:23 +00:00
|
|
|
|
2022-04-07 18:46:23 +00:00
|
|
|
out.State = *power2.ConstructState(em, emm)
|
2021-05-15 01:11:23 +00:00
|
|
|
|
|
|
|
return &out, nil
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
type state2 struct {
|
|
|
|
power2.State
|
2020-09-19 03:46:03 +00:00
|
|
|
store adt.Store
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) TotalLocked() (abi.TokenAmount, error) {
|
2020-09-19 03:46:03 +00:00
|
|
|
return s.TotalPledgeCollateral, nil
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) TotalPower() (Claim, error) {
|
2020-09-19 03:46:03 +00:00
|
|
|
return Claim{
|
|
|
|
RawBytePower: s.TotalRawBytePower,
|
|
|
|
QualityAdjPower: s.TotalQualityAdjPower,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Committed power to the network. Includes miners below the minimum threshold.
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) TotalCommitted() (Claim, error) {
|
2020-09-19 03:46:03 +00:00
|
|
|
return Claim{
|
|
|
|
RawBytePower: s.TotalBytesCommitted,
|
|
|
|
QualityAdjPower: s.TotalQABytesCommitted,
|
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) MinerPower(addr address.Address) (Claim, bool, error) {
|
2021-04-27 18:48:32 +00:00
|
|
|
claims, err := s.claims()
|
2020-09-19 03:46:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return Claim{}, false, err
|
|
|
|
}
|
2020-09-28 20:13:18 +00:00
|
|
|
var claim power2.Claim
|
2020-09-19 03:46:03 +00:00
|
|
|
ok, err := claims.Get(abi.AddrKey(addr), &claim)
|
|
|
|
if err != nil {
|
|
|
|
return Claim{}, false, err
|
|
|
|
}
|
|
|
|
return Claim{
|
|
|
|
RawBytePower: claim.RawBytePower,
|
|
|
|
QualityAdjPower: claim.QualityAdjPower,
|
|
|
|
}, ok, nil
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) MinerNominalPowerMeetsConsensusMinimum(a address.Address) (bool, error) {
|
2020-09-19 03:46:03 +00:00
|
|
|
return s.State.MinerNominalPowerMeetsConsensusMinimum(s.store, a)
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) TotalPowerSmoothed() (builtin.FilterEstimate, error) {
|
|
|
|
return builtin.FromV2FilterEstimate(s.State.ThisEpochQAPowerSmoothed), nil
|
2020-09-19 03:46:03 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) MinerCounts() (uint64, uint64, error) {
|
2020-09-19 03:46:03 +00:00
|
|
|
return uint64(s.State.MinerAboveMinPowerCount), uint64(s.State.MinerCount), nil
|
|
|
|
}
|
|
|
|
|
2020-09-28 20:13:18 +00:00
|
|
|
func (s *state2) ListAllMiners() ([]address.Address, error) {
|
2021-04-27 18:48:32 +00:00
|
|
|
claims, err := s.claims()
|
2020-09-19 03:46:03 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var miners []address.Address
|
|
|
|
err = claims.ForEach(nil, func(k string) error {
|
|
|
|
a, err := address.NewFromBytes([]byte(k))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
miners = append(miners, a)
|
|
|
|
return nil
|
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return miners, nil
|
|
|
|
}
|
2020-10-12 03:31:20 +00:00
|
|
|
|
|
|
|
func (s *state2) ForEachClaim(cb func(miner address.Address, claim Claim) error) error {
|
2021-04-27 18:48:32 +00:00
|
|
|
claims, err := s.claims()
|
2020-10-12 03:31:20 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
var claim power2.Claim
|
|
|
|
return claims.ForEach(&claim, func(k string) error {
|
|
|
|
a, err := address.NewFromBytes([]byte(k))
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
return cb(a, Claim{
|
|
|
|
RawBytePower: claim.RawBytePower,
|
|
|
|
QualityAdjPower: claim.QualityAdjPower,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
}
|
2020-10-28 20:56:02 +00:00
|
|
|
|
|
|
|
func (s *state2) ClaimsChanged(other State) (bool, error) {
|
|
|
|
other2, ok := other.(*state2)
|
|
|
|
if !ok {
|
|
|
|
// treat an upgrade as a change, always
|
|
|
|
return true, nil
|
|
|
|
}
|
|
|
|
return !s.State.Claims.Equals(other2.State.Claims), nil
|
|
|
|
}
|
2020-10-28 23:52:52 +00:00
|
|
|
|
2021-05-15 01:11:23 +00:00
|
|
|
func (s *state2) SetTotalQualityAdjPower(p abi.StoragePower) error {
|
|
|
|
s.State.TotalQualityAdjPower = p
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *state2) SetTotalRawBytePower(p abi.StoragePower) error {
|
|
|
|
s.State.TotalRawBytePower = p
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *state2) SetThisEpochQualityAdjPower(p abi.StoragePower) error {
|
|
|
|
s.State.ThisEpochQualityAdjPower = p
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *state2) SetThisEpochRawBytePower(p abi.StoragePower) error {
|
|
|
|
s.State.ThisEpochRawBytePower = p
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *state2) GetState() interface{} {
|
|
|
|
return &s.State
|
|
|
|
}
|
|
|
|
|
2020-10-28 23:52:52 +00:00
|
|
|
func (s *state2) claims() (adt.Map, error) {
|
|
|
|
return adt2.AsMap(s.store, s.Claims)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *state2) decodeClaim(val *cbg.Deferred) (Claim, error) {
|
|
|
|
var ci power2.Claim
|
|
|
|
if err := ci.UnmarshalCBOR(bytes.NewReader(val.Raw)); err != nil {
|
|
|
|
return Claim{}, err
|
|
|
|
}
|
|
|
|
return fromV2Claim(ci), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func fromV2Claim(v2 power2.Claim) Claim {
|
|
|
|
return Claim{
|
|
|
|
RawBytePower: v2.RawBytePower,
|
|
|
|
QualityAdjPower: v2.QualityAdjPower,
|
|
|
|
}
|
|
|
|
}
|