|
|
|
package params
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math/big"
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Well-known chain IDs.
|
|
|
|
var (
|
|
|
|
MainnetChainID = big.NewInt(1)
|
|
|
|
TestnetChainID = big.NewInt(2)
|
|
|
|
PangaeaChainID = big.NewInt(3)
|
|
|
|
PartnerChainID = big.NewInt(4)
|
|
|
|
StressnetChainID = big.NewInt(5)
|
|
|
|
TestChainID = big.NewInt(99) // not a real network
|
|
|
|
AllProtocolChangesChainID = big.NewInt(100) // not a real network
|
|
|
|
|
|
|
|
// EthMainnetShard0ChainID to be reserved unique chain ID for eth compatible chains.
|
|
|
|
EthMainnetShard0ChainID = big.NewInt(1666600000)
|
|
|
|
EthTestnetShard0ChainID = big.NewInt(1666700000)
|
|
|
|
EthPangaeaShard0ChainID = big.NewInt(1666800000)
|
|
|
|
EthPartnerShard0ChainID = big.NewInt(1666900000)
|
|
|
|
EthStressnetShard0ChainID = big.NewInt(1667000000)
|
|
|
|
EthTestShard0ChainID = big.NewInt(1667100000) // not a real network
|
|
|
|
EthAllProtocolChangesShard0ChainID = big.NewInt(1667200000) // not a real network
|
|
|
|
)
|
|
|
|
|
|
|
|
// EpochTBD is a large, “not anytime soon” epoch. It used as a placeholder
|
|
|
|
// until the exact epoch is decided.
|
|
|
|
var EpochTBD = big.NewInt(10000000)
|
|
|
|
var once sync.Once
|
|
|
|
|
|
|
|
var (
|
|
|
|
// MainnetChainConfig is the chain parameters to run a node on the main network.
|
|
|
|
MainnetChainConfig = &ChainConfig{
|
|
|
|
ChainID: MainnetChainID,
|
|
|
|
EthCompatibleChainID: EthMainnetShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthMainnetShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(442), // Around Thursday Feb 4th 2020, 10AM PST
|
|
|
|
CrossTxEpoch: big.NewInt(28),
|
|
|
|
CrossLinkEpoch: big.NewInt(186),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(689), // Around Wed Sept 15th 2021 with 3.5s block time
|
|
|
|
StakingEpoch: big.NewInt(186),
|
|
|
|
PreStakingEpoch: big.NewInt(185),
|
|
|
|
QuickUnlockEpoch: big.NewInt(191),
|
|
|
|
FiveSecondsEpoch: big.NewInt(230),
|
|
|
|
TwoSecondsEpoch: big.NewInt(366), // Around Tuesday Dec 8th 2020, 8AM PST
|
|
|
|
SixtyPercentEpoch: big.NewInt(530), // Around Monday Apr 12th 2021, 22:30 UTC
|
|
|
|
RedelegationEpoch: big.NewInt(290),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(530), // Around Monday Apr 12th 2021, 22:30 UTC
|
|
|
|
VRFEpoch: big.NewInt(631), // Around Wed July 7th 2021
|
|
|
|
PrevVRFEpoch: big.NewInt(689), // Around Wed Sept 15th 2021 with 3.5s block time
|
|
|
|
MinDelegation100Epoch: big.NewInt(631), // Around Wed July 7th 2021
|
|
|
|
MinCommissionRateEpoch: big.NewInt(631), // Around Wed July 7th 2021
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(100),
|
|
|
|
EPoSBound35Epoch: big.NewInt(631), // Around Wed July 7th 2021
|
|
|
|
EIP155Epoch: big.NewInt(28),
|
|
|
|
S3Epoch: big.NewInt(28),
|
|
|
|
DataCopyFixEpoch: big.NewInt(689), // Around Wed Sept 15th 2021 with 3.5s block time
|
|
|
|
IstanbulEpoch: big.NewInt(314),
|
|
|
|
ReceiptLogEpoch: big.NewInt(101),
|
|
|
|
SHA3Epoch: big.NewInt(725), // Around Mon Oct 11 2021, 19:00 UTC
|
|
|
|
HIP6And8Epoch: big.NewInt(725), // Around Mon Oct 11 2021, 19:00 UTC
|
|
|
|
StakingPrecompileEpoch: big.NewInt(871), // Around Tue Feb 11 2022
|
|
|
|
ChainIdFixEpoch: big.NewInt(1323), // Around Wed 8 Feb 11:30PM UTC
|
|
|
|
SlotsLimitedEpoch: big.NewInt(999), // Around Fri, 27 May 2022 09:41:02 UTC with 2s block time
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(1323), // Around Wed 8 Feb 11:30PM UTC
|
|
|
|
AllowlistEpoch: EpochTBD,
|
|
|
|
FeeCollectEpoch: EpochTBD,
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 64,
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestnetChainConfig contains the chain parameters to run a node on the harmony test network.
|
|
|
|
TestnetChainConfig = &ChainConfig{
|
|
|
|
ChainID: TestnetChainID,
|
|
|
|
EthCompatibleChainID: EthTestnetShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthTestnetShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(0),
|
|
|
|
CrossTxEpoch: big.NewInt(0),
|
|
|
|
CrossLinkEpoch: big.NewInt(2),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(2),
|
|
|
|
StakingEpoch: big.NewInt(2),
|
|
|
|
PreStakingEpoch: big.NewInt(1),
|
|
|
|
QuickUnlockEpoch: big.NewInt(0),
|
|
|
|
FiveSecondsEpoch: big.NewInt(0),
|
|
|
|
TwoSecondsEpoch: big.NewInt(2),
|
|
|
|
SixtyPercentEpoch: big.NewInt(2),
|
|
|
|
RedelegationEpoch: big.NewInt(2),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(2),
|
|
|
|
VRFEpoch: big.NewInt(2),
|
|
|
|
PrevVRFEpoch: big.NewInt(2),
|
|
|
|
MinDelegation100Epoch: big.NewInt(2),
|
|
|
|
MinCommissionRateEpoch: big.NewInt(2),
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(2),
|
|
|
|
EPoSBound35Epoch: big.NewInt(2),
|
|
|
|
EIP155Epoch: big.NewInt(0),
|
|
|
|
S3Epoch: big.NewInt(0),
|
|
|
|
DataCopyFixEpoch: big.NewInt(0),
|
|
|
|
IstanbulEpoch: big.NewInt(0),
|
|
|
|
ReceiptLogEpoch: big.NewInt(0),
|
|
|
|
SHA3Epoch: big.NewInt(0),
|
|
|
|
HIP6And8Epoch: big.NewInt(2),
|
|
|
|
StakingPrecompileEpoch: big.NewInt(2),
|
|
|
|
SlotsLimitedEpoch: big.NewInt(2),
|
|
|
|
ChainIdFixEpoch: big.NewInt(0),
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(2),
|
|
|
|
AllowlistEpoch: big.NewInt(2),
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 64,
|
|
|
|
FeeCollectEpoch: EpochTBD,
|
|
|
|
}
|
|
|
|
// PangaeaChainConfig contains the chain parameters for the Pangaea network.
|
|
|
|
// All features except for CrossLink are enabled at launch.
|
|
|
|
PangaeaChainConfig = &ChainConfig{
|
|
|
|
ChainID: PangaeaChainID,
|
|
|
|
EthCompatibleChainID: EthPangaeaShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthPangaeaShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(0),
|
|
|
|
CrossTxEpoch: big.NewInt(0),
|
|
|
|
CrossLinkEpoch: big.NewInt(2),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(3),
|
|
|
|
StakingEpoch: big.NewInt(2),
|
|
|
|
PreStakingEpoch: big.NewInt(1),
|
|
|
|
QuickUnlockEpoch: big.NewInt(0),
|
|
|
|
FiveSecondsEpoch: big.NewInt(0),
|
|
|
|
TwoSecondsEpoch: big.NewInt(0),
|
|
|
|
SixtyPercentEpoch: big.NewInt(0),
|
|
|
|
RedelegationEpoch: big.NewInt(0),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(0),
|
|
|
|
VRFEpoch: big.NewInt(0),
|
|
|
|
PrevVRFEpoch: big.NewInt(0),
|
|
|
|
MinDelegation100Epoch: big.NewInt(0),
|
|
|
|
MinCommissionRateEpoch: big.NewInt(0),
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(10),
|
|
|
|
EPoSBound35Epoch: big.NewInt(0),
|
|
|
|
EIP155Epoch: big.NewInt(0),
|
|
|
|
S3Epoch: big.NewInt(0),
|
|
|
|
DataCopyFixEpoch: big.NewInt(0),
|
|
|
|
IstanbulEpoch: big.NewInt(0),
|
|
|
|
ReceiptLogEpoch: big.NewInt(0),
|
|
|
|
SHA3Epoch: big.NewInt(0),
|
|
|
|
HIP6And8Epoch: big.NewInt(0),
|
|
|
|
StakingPrecompileEpoch: big.NewInt(2), // same as staking
|
|
|
|
ChainIdFixEpoch: big.NewInt(0),
|
|
|
|
SlotsLimitedEpoch: EpochTBD, // epoch to enable HIP-16
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(1),
|
|
|
|
AllowlistEpoch: EpochTBD,
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 64,
|
|
|
|
FeeCollectEpoch: EpochTBD,
|
|
|
|
}
|
|
|
|
|
|
|
|
// PartnerChainConfig contains the chain parameters for the Partner network.
|
|
|
|
// This is the Devnet config
|
|
|
|
PartnerChainConfig = &ChainConfig{
|
|
|
|
ChainID: PartnerChainID,
|
|
|
|
EthCompatibleChainID: EthPartnerShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthPartnerShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(0),
|
|
|
|
CrossTxEpoch: big.NewInt(0),
|
|
|
|
CrossLinkEpoch: big.NewInt(2),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(3),
|
|
|
|
StakingEpoch: big.NewInt(2),
|
|
|
|
PreStakingEpoch: big.NewInt(1),
|
|
|
|
QuickUnlockEpoch: big.NewInt(0),
|
|
|
|
FiveSecondsEpoch: big.NewInt(0),
|
|
|
|
TwoSecondsEpoch: big.NewInt(0),
|
|
|
|
SixtyPercentEpoch: big.NewInt(4),
|
|
|
|
RedelegationEpoch: big.NewInt(0),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(0),
|
|
|
|
VRFEpoch: big.NewInt(0),
|
|
|
|
PrevVRFEpoch: big.NewInt(0),
|
|
|
|
MinDelegation100Epoch: big.NewInt(0),
|
|
|
|
MinCommissionRateEpoch: big.NewInt(0),
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(10),
|
|
|
|
EPoSBound35Epoch: big.NewInt(0),
|
|
|
|
EIP155Epoch: big.NewInt(0),
|
|
|
|
S3Epoch: big.NewInt(0),
|
|
|
|
DataCopyFixEpoch: big.NewInt(0),
|
|
|
|
IstanbulEpoch: big.NewInt(0),
|
|
|
|
ReceiptLogEpoch: big.NewInt(0),
|
|
|
|
SHA3Epoch: big.NewInt(0),
|
|
|
|
HIP6And8Epoch: big.NewInt(0),
|
|
|
|
StakingPrecompileEpoch: big.NewInt(2),
|
|
|
|
ChainIdFixEpoch: big.NewInt(0),
|
|
|
|
SlotsLimitedEpoch: EpochTBD, // epoch to enable HIP-16
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(1),
|
|
|
|
AllowlistEpoch: EpochTBD,
|
|
|
|
FeeCollectEpoch: big.NewInt(574),
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 64,
|
|
|
|
}
|
|
|
|
|
|
|
|
// StressnetChainConfig contains the chain parameters for the Stress test network.
|
|
|
|
// All features except for CrossLink are enabled at launch.
|
|
|
|
StressnetChainConfig = &ChainConfig{
|
|
|
|
ChainID: StressnetChainID,
|
|
|
|
EthCompatibleChainID: EthStressnetShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthStressnetShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(0),
|
|
|
|
CrossTxEpoch: big.NewInt(0),
|
|
|
|
CrossLinkEpoch: big.NewInt(2),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(3),
|
|
|
|
StakingEpoch: big.NewInt(2),
|
|
|
|
PreStakingEpoch: big.NewInt(1),
|
|
|
|
QuickUnlockEpoch: big.NewInt(0),
|
|
|
|
FiveSecondsEpoch: big.NewInt(0),
|
|
|
|
TwoSecondsEpoch: big.NewInt(0),
|
|
|
|
SixtyPercentEpoch: big.NewInt(10),
|
|
|
|
RedelegationEpoch: big.NewInt(0),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(0),
|
|
|
|
VRFEpoch: big.NewInt(0),
|
|
|
|
PrevVRFEpoch: big.NewInt(0),
|
|
|
|
MinDelegation100Epoch: big.NewInt(0),
|
|
|
|
MinCommissionRateEpoch: big.NewInt(0),
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(10),
|
|
|
|
EPoSBound35Epoch: big.NewInt(0),
|
|
|
|
EIP155Epoch: big.NewInt(0),
|
|
|
|
S3Epoch: big.NewInt(0),
|
|
|
|
DataCopyFixEpoch: big.NewInt(0),
|
|
|
|
IstanbulEpoch: big.NewInt(0),
|
|
|
|
ReceiptLogEpoch: big.NewInt(0),
|
|
|
|
SHA3Epoch: big.NewInt(0),
|
|
|
|
HIP6And8Epoch: big.NewInt(0),
|
|
|
|
StakingPrecompileEpoch: big.NewInt(2),
|
|
|
|
ChainIdFixEpoch: big.NewInt(0),
|
|
|
|
SlotsLimitedEpoch: EpochTBD, // epoch to enable HIP-16
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(1),
|
|
|
|
AllowlistEpoch: EpochTBD,
|
|
|
|
FeeCollectEpoch: EpochTBD,
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 64,
|
|
|
|
}
|
|
|
|
|
|
|
|
// LocalnetChainConfig contains the chain parameters to run for local development.
|
|
|
|
LocalnetChainConfig = &ChainConfig{
|
|
|
|
ChainID: TestnetChainID,
|
|
|
|
EthCompatibleChainID: EthTestnetShard0ChainID,
|
|
|
|
EthCompatibleShard0ChainID: EthTestnetShard0ChainID,
|
|
|
|
EthCompatibleEpoch: big.NewInt(0),
|
|
|
|
CrossTxEpoch: big.NewInt(0),
|
|
|
|
CrossLinkEpoch: big.NewInt(2),
|
|
|
|
AggregatedRewardEpoch: big.NewInt(3),
|
|
|
|
StakingEpoch: big.NewInt(2),
|
|
|
|
PreStakingEpoch: big.NewInt(0),
|
|
|
|
QuickUnlockEpoch: big.NewInt(0),
|
|
|
|
FiveSecondsEpoch: big.NewInt(0),
|
|
|
|
TwoSecondsEpoch: big.NewInt(0),
|
|
|
|
SixtyPercentEpoch: EpochTBD, // Never enable it for localnet as localnet has no external validator setup
|
|
|
|
RedelegationEpoch: big.NewInt(0),
|
|
|
|
NoEarlyUnlockEpoch: big.NewInt(0),
|
|
|
|
VRFEpoch: big.NewInt(0),
|
|
|
|
PrevVRFEpoch: big.NewInt(0),
|
|
|
|
MinDelegation100Epoch: big.NewInt(0),
|
|
|
|
MinCommissionRateEpoch: big.NewInt(0),
|
|
|
|
MinCommissionPromoPeriod: big.NewInt(10),
|
|
|
|
EPoSBound35Epoch: big.NewInt(0),
|
|
|
|
EIP155Epoch: big.NewInt(0),
|
|
|
|
S3Epoch: big.NewInt(0),
|
|
|
|
DataCopyFixEpoch: big.NewInt(0),
|
|
|
|
IstanbulEpoch: big.NewInt(0),
|
|
|
|
ReceiptLogEpoch: big.NewInt(0),
|
|
|
|
SHA3Epoch: big.NewInt(0),
|
|
|
|
HIP6And8Epoch: EpochTBD, // Never enable it for localnet as localnet has no external validator setup
|
|
|
|
StakingPrecompileEpoch: big.NewInt(2),
|
|
|
|
ChainIdFixEpoch: big.NewInt(0),
|
|
|
|
SlotsLimitedEpoch: EpochTBD, // epoch to enable HIP-16
|
|
|
|
CrossShardXferPrecompileEpoch: big.NewInt(1),
|
|
|
|
AllowlistEpoch: EpochTBD,
|
|
|
|
FeeCollectEpoch: big.NewInt(5),
|
|
|
|
LeaderRotationEpoch: EpochTBD,
|
|
|
|
LeaderRotationBlocksCount: 5,
|
|
|
|
}
|
|
|
|
|
|
|
|
// AllProtocolChanges ...
|
|
|
|
// This configuration is intentionally not using keyed fields to force anyone
|
|
|
|
// adding flags to the config to also have to set these fields.
|
|
|
|
AllProtocolChanges = &ChainConfig{
|
|
|
|
AllProtocolChangesChainID, // ChainID
|
|
|
|
EthAllProtocolChangesShard0ChainID, // EthCompatibleChainID
|
|
|
|
EthAllProtocolChangesShard0ChainID, // EthCompatibleShard0ChainID
|
|
|
|
big.NewInt(0), // EthCompatibleEpoch
|
|
|
|
big.NewInt(0), // CrossTxEpoch
|
|
|
|
big.NewInt(0), // CrossLinkEpoch
|
|
|
|
big.NewInt(0), // AggregatedRewardEpoch
|
|
|
|
big.NewInt(0), // StakingEpoch
|
|
|
|
big.NewInt(0), // PreStakingEpoch
|
|
|
|
big.NewInt(0), // QuickUnlockEpoch
|
|
|
|
big.NewInt(0), // FiveSecondsEpoch
|
|
|
|
big.NewInt(0), // TwoSecondsEpoch
|
|
|
|
big.NewInt(0), // SixtyPercentEpoch
|
|
|
|
big.NewInt(0), // RedelegationEpoch
|
|
|
|
big.NewInt(0), // NoEarlyUnlockEpoch
|
|
|
|
big.NewInt(0), // VRFEpoch
|
|
|
|
big.NewInt(0), // PrevVRFEpoch
|
|
|
|
big.NewInt(0), // MinDelegation100Epoch
|
|
|
|
big.NewInt(0), // MinCommissionRateEpoch
|
|
|
|
big.NewInt(10), // MinCommissionPromoPeriod
|
|
|
|
big.NewInt(0), // EPoSBound35Epoch
|
|
|
|
big.NewInt(0), // EIP155Epoch
|
|
|
|
big.NewInt(0), // S3Epoch
|
|
|
|
big.NewInt(0), // DataCopyFixEpoch
|
|
|
|
big.NewInt(0), // IstanbulEpoch
|
|
|
|
big.NewInt(0), // ReceiptLogEpoch
|
|
|
|
big.NewInt(0), // SHA3Epoch
|
|
|
|
big.NewInt(0), // HIP6And8Epoch
|
Resolve harmony-one/bounties#77: Staking precompiles (#3906)
* Resolve harmony-one/bounties#77: Staking precompiles
Create write capable precompiles that can perform staking transactions
Add hard fork logic (EpochTBD) for these precompiles
Tests for new code with at least 80% unit test coverage
Staking library + tests in MaxMustermann2/harmony-staking-precompiles
* Fix small typo in comment
* Run goimports on files to fix Travis
* Do not activate staking precompile on shard 0
* Cascade readOnly to WriteCapableContract
* No overlap in readOnly + writeCapable precompiles
* Use function selector instead of directive
From Solidity, use abi.encodeWithSelector and match it against the
exact ABI of the functions. This allows us to remove the need for
a directive (32) being encoded, and thus saves 28 bytes of data.
* Do not allow contracts to become validators
As discussed with Jacky on #3906
* Merge harmony-one/harmony/main properly this time
* Run goimports
* Update gas calculation for staking precompile
Please see comment in core/vm/contracts_write.go RequiredGas
* Do not allow contract to become validator (2/2)
* Cache StakeMsgs from precompiled transactions
Add the StakeMsgs to ProcessorResult and cascade them in insertChain
* Remove ContractCode fields from validators
Since smart contracts can no longer beecome validators,
this field is superfluous. Remove it from the Wrapper
structure, and do not assign it a value when creating
a validator. Build and goimports checked
* Update comments in response to feedback
(1) Comments to start with function names
(2) Comments for public variables
(3) Comment to match function name RunPrecompiledContract
(4) Clarify that CreateValidatorFunc + EditValidatorFunc are still used
* Fix Travis build by reverting rosetta change
* Add revert capability to 3 staking tx types
- Delegate
- Undelegate
- CollectRewards
* Fix build: Update evm_test for ValidatorWrapper
* Merge main into harmony-staking-precompiles
* Add gas for precompile calls and allow EOA usage
- Each time the precompile is called, charge the base gas fee plus data
cost (if data can be parsed successfully). A gas fee is added to
prevent benevolent contract deployers from subsidizing the staking
transactions for EOAs through repeated assembly `delegatecall`.
- Allow EOAs to use the staking precompile directly. Some changes to
the Solidity library are associated with this change.
- Remove bytes from parsing address, since the ABI unpacks it into an
address format correctly.
- Add or update tests. Test coverage report to be attached to the PR
shortly.
* Run goimports
* Check read only and write capable for overlap
* Handle precompile stakeMsgs for block proposer
The staking precompile generates staking messages which are cascaded to
the block via the EVM in `state_processor.go`. This change cascades them
in `worker.go` to allow block proposers and block verifiers to keep the
same state.
* Run goimports for cf2dfac4081444e36a120c9432f4e..
* Update staking precompile epoch to 2 for localnet
Bring it in line with staking epoch. Change effects all configurations
except mainnet and testnet. `goimports` included.
* Add read only precompile to fetch the epoch num
* Move epoch precompile to 250
* precompiles: left pad the returned epoch number
* chainConfig: check epochs for precompiles
panic if staking precompile epoch < pre staking epoch
* Add staking migration precompile
- Lives at address 251
- Migrates delegations + pending undelegations from address A to B
- Useful if address A is hacked
- Charges gas of 21k + cost of bytes for two addresses
- Does not remove existing delegations, just sets them to zero.
Replicates current undelegate setup
- Unit tests and `goimports` included. Integration test following
shortly in MaxMustermann2/harmony-staking-precompiles
* Migration precompile: merge into staking
Merge the two precompiles into one, add gas calculation for migration
precompile. Move epoch precompile to 251 as a result. When migrating,
add undelegations to `To`'s existing undelegations, if any match the
epoch.
* Add migration gas test, remove panic, add check
In response to review comments, add tests for migration gas wherein
there are 0/1/2 delegations to migrate. Add the index out of bound check
to migration gas calculator and remove panics. Lastly, re-sort
migrated undelegations if no existing undelegation in the same epoch was
found on `To`.
* Move undelegations sorting to end of loop
3 years ago
|
|
|
big.NewInt(0), // StakingPrecompileEpoch
|
|
|
|
big.NewInt(0), // ChainIdFixEpoch
|
|
|
|
big.NewInt(0), // SlotsLimitedEpoch
|
|
|
|
big.NewInt(1), // CrossShardXferPrecompileEpoch
|
|
|
|
big.NewInt(0), // AllowlistEpoch
|
|
|
|
big.NewInt(1), // LeaderRotationEpoch
|
|
|
|
64, // LeaderRotationBlocksCount
|
|
|
|
big.NewInt(0), // FeeCollectEpoch
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestChainConfig ...
|
|
|
|
// This configuration is intentionally not using keyed fields to force anyone
|
|
|
|
// adding flags to the config to also have to set these fields.
|
|
|
|
TestChainConfig = &ChainConfig{
|
|
|
|
TestChainID, // ChainID
|
|
|
|
EthTestShard0ChainID, // EthCompatibleChainID
|
|
|
|
EthTestShard0ChainID, // EthCompatibleShard0ChainID
|
|
|
|
big.NewInt(0), // EthCompatibleEpoch
|
|
|
|
big.NewInt(0), // CrossTxEpoch
|
|
|
|
big.NewInt(0), // CrossLinkEpoch
|
|
|
|
big.NewInt(0), // AggregatedRewardEpoch
|
|
|
|
big.NewInt(0), // StakingEpoch
|
|
|
|
big.NewInt(0), // PreStakingEpoch
|
|
|
|
big.NewInt(0), // QuickUnlockEpoch
|
|
|
|
big.NewInt(0), // FiveSecondsEpoch
|
|
|
|
big.NewInt(0), // TwoSecondsEpoch
|
|
|
|
big.NewInt(0), // SixtyPercentEpoch
|
|
|
|
big.NewInt(0), // RedelegationEpoch
|
|
|
|
big.NewInt(0), // NoEarlyUnlockEpoch
|
|
|
|
big.NewInt(0), // VRFEpoch
|
|
|
|
big.NewInt(0), // PrevVRFEpoch
|
|
|
|
big.NewInt(0), // MinDelegation100Epoch
|
|
|
|
big.NewInt(0), // MinCommissionRateEpoch
|
|
|
|
big.NewInt(10), // MinCommissionPromoPeriod
|
|
|
|
big.NewInt(0), // EPoSBound35Epoch
|
|
|
|
big.NewInt(0), // EIP155Epoch
|
|
|
|
big.NewInt(0), // S3Epoch
|
|
|
|
big.NewInt(0), // DataCopyFixEpoch
|
|
|
|
big.NewInt(0), // IstanbulEpoch
|
|
|
|
big.NewInt(0), // ReceiptLogEpoch
|
|
|
|
big.NewInt(0), // SHA3Epoch
|
|
|
|
big.NewInt(0), // HIP6And8Epoch
|
Resolve harmony-one/bounties#77: Staking precompiles (#3906)
* Resolve harmony-one/bounties#77: Staking precompiles
Create write capable precompiles that can perform staking transactions
Add hard fork logic (EpochTBD) for these precompiles
Tests for new code with at least 80% unit test coverage
Staking library + tests in MaxMustermann2/harmony-staking-precompiles
* Fix small typo in comment
* Run goimports on files to fix Travis
* Do not activate staking precompile on shard 0
* Cascade readOnly to WriteCapableContract
* No overlap in readOnly + writeCapable precompiles
* Use function selector instead of directive
From Solidity, use abi.encodeWithSelector and match it against the
exact ABI of the functions. This allows us to remove the need for
a directive (32) being encoded, and thus saves 28 bytes of data.
* Do not allow contracts to become validators
As discussed with Jacky on #3906
* Merge harmony-one/harmony/main properly this time
* Run goimports
* Update gas calculation for staking precompile
Please see comment in core/vm/contracts_write.go RequiredGas
* Do not allow contract to become validator (2/2)
* Cache StakeMsgs from precompiled transactions
Add the StakeMsgs to ProcessorResult and cascade them in insertChain
* Remove ContractCode fields from validators
Since smart contracts can no longer beecome validators,
this field is superfluous. Remove it from the Wrapper
structure, and do not assign it a value when creating
a validator. Build and goimports checked
* Update comments in response to feedback
(1) Comments to start with function names
(2) Comments for public variables
(3) Comment to match function name RunPrecompiledContract
(4) Clarify that CreateValidatorFunc + EditValidatorFunc are still used
* Fix Travis build by reverting rosetta change
* Add revert capability to 3 staking tx types
- Delegate
- Undelegate
- CollectRewards
* Fix build: Update evm_test for ValidatorWrapper
* Merge main into harmony-staking-precompiles
* Add gas for precompile calls and allow EOA usage
- Each time the precompile is called, charge the base gas fee plus data
cost (if data can be parsed successfully). A gas fee is added to
prevent benevolent contract deployers from subsidizing the staking
transactions for EOAs through repeated assembly `delegatecall`.
- Allow EOAs to use the staking precompile directly. Some changes to
the Solidity library are associated with this change.
- Remove bytes from parsing address, since the ABI unpacks it into an
address format correctly.
- Add or update tests. Test coverage report to be attached to the PR
shortly.
* Run goimports
* Check read only and write capable for overlap
* Handle precompile stakeMsgs for block proposer
The staking precompile generates staking messages which are cascaded to
the block via the EVM in `state_processor.go`. This change cascades them
in `worker.go` to allow block proposers and block verifiers to keep the
same state.
* Run goimports for cf2dfac4081444e36a120c9432f4e..
* Update staking precompile epoch to 2 for localnet
Bring it in line with staking epoch. Change effects all configurations
except mainnet and testnet. `goimports` included.
* Add read only precompile to fetch the epoch num
* Move epoch precompile to 250
* precompiles: left pad the returned epoch number
* chainConfig: check epochs for precompiles
panic if staking precompile epoch < pre staking epoch
* Add staking migration precompile
- Lives at address 251
- Migrates delegations + pending undelegations from address A to B
- Useful if address A is hacked
- Charges gas of 21k + cost of bytes for two addresses
- Does not remove existing delegations, just sets them to zero.
Replicates current undelegate setup
- Unit tests and `goimports` included. Integration test following
shortly in MaxMustermann2/harmony-staking-precompiles
* Migration precompile: merge into staking
Merge the two precompiles into one, add gas calculation for migration
precompile. Move epoch precompile to 251 as a result. When migrating,
add undelegations to `To`'s existing undelegations, if any match the
epoch.
* Add migration gas test, remove panic, add check
In response to review comments, add tests for migration gas wherein
there are 0/1/2 delegations to migrate. Add the index out of bound check
to migration gas calculator and remove panics. Lastly, re-sort
migrated undelegations if no existing undelegation in the same epoch was
found on `To`.
* Move undelegations sorting to end of loop
3 years ago
|
|
|
big.NewInt(0), // StakingPrecompileEpoch
|
|
|
|
big.NewInt(0), // ChainIdFixEpoch
|
|
|
|
big.NewInt(0), // SlotsLimitedEpoch
|
|
|
|
big.NewInt(1), // CrossShardXferPrecompileEpoch
|
|
|
|
big.NewInt(0), // AllowlistEpoch
|
|
|
|
big.NewInt(1), // LeaderRotationEpoch
|
|
|
|
64, // LeaderRotationBlocksCount
|
|
|
|
big.NewInt(0), // FeeCollectEpoch
|
|
|
|
}
|
|
|
|
|
|
|
|
// TestRules ...
|
|
|
|
TestRules = TestChainConfig.Rules(new(big.Int))
|
|
|
|
)
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
MainnetChainConfig.mustValid()
|
|
|
|
TestnetChainConfig.mustValid()
|
|
|
|
PangaeaChainConfig.mustValid()
|
|
|
|
PartnerChainConfig.mustValid()
|
|
|
|
StressnetChainConfig.mustValid()
|
|
|
|
LocalnetChainConfig.mustValid()
|
|
|
|
}
|
|
|
|
|
|
|
|
// TrustedCheckpoint represents a set of post-processed trie roots (CHT and
|
|
|
|
// BloomTrie) associated with the appropriate section index and head hash. It is
|
|
|
|
// used to start light syncing from this checkpoint and avoid downloading the
|
|
|
|
// entire header chain while still being able to securely access old headers/logs.
|
|
|
|
type TrustedCheckpoint struct {
|
|
|
|
Name string `json:"-"`
|
|
|
|
SectionIndex uint64 `json:"sectionIndex"`
|
|
|
|
SectionHead common.Hash `json:"sectionHead"`
|
|
|
|
CHTRoot common.Hash `json:"chtRoot"`
|
|
|
|
BloomRoot common.Hash `json:"bloomRoot"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// ChainConfig is the core config which determines the blockchain settings.
|
|
|
|
//
|
|
|
|
// ChainConfig is stored in the database on a per block basis. This means
|
|
|
|
// that any network, identified by its genesis block, can have its own
|
|
|
|
// set of configuration options.
|
|
|
|
type ChainConfig struct {
|
|
|
|
// ChainId identifies the current chain and is used for replay protection
|
|
|
|
ChainID *big.Int `json:"chain-id"`
|
|
|
|
|
|
|
|
// EthCompatibleChainID identifies the chain id used for ethereum compatible transactions
|
|
|
|
EthCompatibleChainID *big.Int `json:"eth-compatible-chain-id"`
|
|
|
|
|
|
|
|
// EthCompatibleShard0ChainID identifies the shard 0 chain id used for ethereum compatible transactions
|
|
|
|
EthCompatibleShard0ChainID *big.Int `json:"eth-compatible-shard-0-chain-id"`
|
|
|
|
|
|
|
|
// EthCompatibleEpoch is the epoch where ethereum-compatible transaction starts being
|
|
|
|
// processed.
|
|
|
|
EthCompatibleEpoch *big.Int `json:"eth-compatible-epoch,omitempty"`
|
|
|
|
|
|
|
|
// CrossTxEpoch is the epoch where cross-shard transaction starts being
|
|
|
|
// processed.
|
|
|
|
CrossTxEpoch *big.Int `json:"cross-tx-epoch,omitempty"`
|
|
|
|
|
|
|
|
// CrossLinkEpoch is the epoch where beaconchain starts containing
|
|
|
|
// cross-shard links.
|
|
|
|
CrossLinkEpoch *big.Int `json:"cross-link-epoch,omitempty"`
|
|
|
|
|
|
|
|
// AggregatedRewardEpoch is the epoch when block rewards are distributed every 64 blocks
|
|
|
|
AggregatedRewardEpoch *big.Int `json:"aggregated-reward-epoch,omitempty"`
|
|
|
|
|
|
|
|
// StakingEpoch is the epoch when shard assign takes staking into account
|
|
|
|
StakingEpoch *big.Int `json:"staking-epoch,omitempty"`
|
|
|
|
|
|
|
|
// PreStakingEpoch is the epoch we allow staking transactions
|
|
|
|
PreStakingEpoch *big.Int `json:"prestaking-epoch,omitempty"`
|
|
|
|
|
|
|
|
// QuickUnlockEpoch is the epoch when undelegation will be unlocked at the current epoch
|
|
|
|
QuickUnlockEpoch *big.Int `json:"quick-unlock-epoch,omitempty"`
|
|
|
|
|
|
|
|
// FiveSecondsEpoch is the epoch when block time is reduced to 5 seconds
|
|
|
|
// and block rewards adjusted to 17.5 ONE/block
|
|
|
|
FiveSecondsEpoch *big.Int `json:"five-seconds-epoch,omitempty"`
|
|
|
|
|
|
|
|
// TwoSecondsEpoch is the epoch when block time is reduced to 2 seconds
|
|
|
|
// and block rewards adjusted to 7 ONE/block
|
|
|
|
TwoSecondsEpoch *big.Int `json:"two-seconds-epoch,omitempty"`
|
|
|
|
|
|
|
|
// SixtyPercentEpoch is the epoch when internal voting power reduced from 68% to 60%
|
|
|
|
SixtyPercentEpoch *big.Int `json:"sixty-percent-epoch,omitempty"`
|
|
|
|
|
|
|
|
// RedelegationEpoch is the epoch when redelegation is supported and undelegation locking time
|
|
|
|
// is restored to 7 epoch
|
|
|
|
RedelegationEpoch *big.Int `json:"redelegation-epoch,omitempty"`
|
|
|
|
|
|
|
|
// NoEarlyUnlockEpoch is the epoch when the early unlock of undelegated token from validators who were elected for
|
|
|
|
// more than 7 epochs is disabled
|
|
|
|
NoEarlyUnlockEpoch *big.Int `json:"no-early-unlock-epoch,omitempty"`
|
|
|
|
|
|
|
|
// VRFEpoch is the epoch when VRF randomness is enabled
|
|
|
|
VRFEpoch *big.Int `json:"vrf-epoch,omitempty"`
|
|
|
|
|
|
|
|
// PrevVRFEpoch is the epoch when previous VRF randomness can be fetched
|
|
|
|
PrevVRFEpoch *big.Int `json:"prev-vrf-epoch,omitempty"`
|
|
|
|
|
|
|
|
// MinDelegation100Epoch is the epoch when min delegation is reduced from 1000 ONE to 100 ONE
|
|
|
|
MinDelegation100Epoch *big.Int `json:"min-delegation-100-epoch,omitempty"`
|
|
|
|
|
|
|
|
// MinCommissionRateEpoch is the epoch when policy for minimum comission rate of 5% is started
|
|
|
|
MinCommissionRateEpoch *big.Int `json:"min-commission-rate-epoch,omitempty"`
|
|
|
|
|
|
|
|
// MinCommissionPromoPeriod is the number of epochs when newly elected validators can have 0% commission
|
|
|
|
MinCommissionPromoPeriod *big.Int `json:"commission-promo-period,omitempty"`
|
|
|
|
|
|
|
|
// EPoSBound35Epoch is the epoch when the EPoS bound parameter c is changed from 15% to 35%
|
|
|
|
EPoSBound35Epoch *big.Int `json:"epos-bound-35-epoch,omitempty"`
|
|
|
|
|
|
|
|
// EIP155 hard fork epoch (include EIP158 too)
|
|
|
|
EIP155Epoch *big.Int `json:"eip155-epoch,omitempty"`
|
|
|
|
|
|
|
|
// S3 epoch is the first epoch containing S3 mainnet and all ethereum update up to Constantinople
|
|
|
|
S3Epoch *big.Int `json:"s3-epoch,omitempty"`
|
|
|
|
|
|
|
|
// DataCopyFix epoch is the first epoch containing fix for evm datacopy bug.
|
|
|
|
DataCopyFixEpoch *big.Int `json:"data-copy-fix-epoch,omitempty"`
|
|
|
|
|
|
|
|
// Istanbul epoch
|
|
|
|
IstanbulEpoch *big.Int `json:"istanbul-epoch,omitempty"`
|
|
|
|
|
|
|
|
// ReceiptLogEpoch is the first epoch support receiptlog
|
|
|
|
ReceiptLogEpoch *big.Int `json:"receipt-log-epoch,omitempty"`
|
|
|
|
|
|
|
|
// IsSHA3Epoch is the first epoch in supporting SHA3 FIPS-202 standard
|
|
|
|
SHA3Epoch *big.Int `json:"sha3-epoch,omitempty"`
|
|
|
|
|
|
|
|
// IsHIP6And8Epoch is the first epoch to support HIP-6 and HIP-8
|
|
|
|
HIP6And8Epoch *big.Int `json:"hip6_8-epoch,omitempty"`
|
Resolve harmony-one/bounties#77: Staking precompiles (#3906)
* Resolve harmony-one/bounties#77: Staking precompiles
Create write capable precompiles that can perform staking transactions
Add hard fork logic (EpochTBD) for these precompiles
Tests for new code with at least 80% unit test coverage
Staking library + tests in MaxMustermann2/harmony-staking-precompiles
* Fix small typo in comment
* Run goimports on files to fix Travis
* Do not activate staking precompile on shard 0
* Cascade readOnly to WriteCapableContract
* No overlap in readOnly + writeCapable precompiles
* Use function selector instead of directive
From Solidity, use abi.encodeWithSelector and match it against the
exact ABI of the functions. This allows us to remove the need for
a directive (32) being encoded, and thus saves 28 bytes of data.
* Do not allow contracts to become validators
As discussed with Jacky on #3906
* Merge harmony-one/harmony/main properly this time
* Run goimports
* Update gas calculation for staking precompile
Please see comment in core/vm/contracts_write.go RequiredGas
* Do not allow contract to become validator (2/2)
* Cache StakeMsgs from precompiled transactions
Add the StakeMsgs to ProcessorResult and cascade them in insertChain
* Remove ContractCode fields from validators
Since smart contracts can no longer beecome validators,
this field is superfluous. Remove it from the Wrapper
structure, and do not assign it a value when creating
a validator. Build and goimports checked
* Update comments in response to feedback
(1) Comments to start with function names
(2) Comments for public variables
(3) Comment to match function name RunPrecompiledContract
(4) Clarify that CreateValidatorFunc + EditValidatorFunc are still used
* Fix Travis build by reverting rosetta change
* Add revert capability to 3 staking tx types
- Delegate
- Undelegate
- CollectRewards
* Fix build: Update evm_test for ValidatorWrapper
* Merge main into harmony-staking-precompiles
* Add gas for precompile calls and allow EOA usage
- Each time the precompile is called, charge the base gas fee plus data
cost (if data can be parsed successfully). A gas fee is added to
prevent benevolent contract deployers from subsidizing the staking
transactions for EOAs through repeated assembly `delegatecall`.
- Allow EOAs to use the staking precompile directly. Some changes to
the Solidity library are associated with this change.
- Remove bytes from parsing address, since the ABI unpacks it into an
address format correctly.
- Add or update tests. Test coverage report to be attached to the PR
shortly.
* Run goimports
* Check read only and write capable for overlap
* Handle precompile stakeMsgs for block proposer
The staking precompile generates staking messages which are cascaded to
the block via the EVM in `state_processor.go`. This change cascades them
in `worker.go` to allow block proposers and block verifiers to keep the
same state.
* Run goimports for cf2dfac4081444e36a120c9432f4e..
* Update staking precompile epoch to 2 for localnet
Bring it in line with staking epoch. Change effects all configurations
except mainnet and testnet. `goimports` included.
* Add read only precompile to fetch the epoch num
* Move epoch precompile to 250
* precompiles: left pad the returned epoch number
* chainConfig: check epochs for precompiles
panic if staking precompile epoch < pre staking epoch
* Add staking migration precompile
- Lives at address 251
- Migrates delegations + pending undelegations from address A to B
- Useful if address A is hacked
- Charges gas of 21k + cost of bytes for two addresses
- Does not remove existing delegations, just sets them to zero.
Replicates current undelegate setup
- Unit tests and `goimports` included. Integration test following
shortly in MaxMustermann2/harmony-staking-precompiles
* Migration precompile: merge into staking
Merge the two precompiles into one, add gas calculation for migration
precompile. Move epoch precompile to 251 as a result. When migrating,
add undelegations to `To`'s existing undelegations, if any match the
epoch.
* Add migration gas test, remove panic, add check
In response to review comments, add tests for migration gas wherein
there are 0/1/2 delegations to migrate. Add the index out of bound check
to migration gas calculator and remove panics. Lastly, re-sort
migrated undelegations if no existing undelegation in the same epoch was
found on `To`.
* Move undelegations sorting to end of loop
3 years ago
|
|
|
|
|
|
|
// StakingPrecompileEpoch is the first epoch to support the staking precompiles
|
|
|
|
StakingPrecompileEpoch *big.Int `json:"staking-precompile-epoch,omitempty"`
|
|
|
|
|
|
|
|
// ChainIdFixEpoch is the first epoch to return ethereum compatible chain id by ChainID() op code
|
|
|
|
ChainIdFixEpoch *big.Int `json:"chain-id-fix-epoch,omitempty"`
|
|
|
|
|
|
|
|
// SlotsLimitedEpoch is the first epoch to enable HIP-16.
|
|
|
|
SlotsLimitedEpoch *big.Int `json:"slots-limit-epoch,omitempty"`
|
|
|
|
|
|
|
|
// CrossShardXferPrecompileEpoch is the first epoch to feature cross shard transfer precompile
|
|
|
|
CrossShardXferPrecompileEpoch *big.Int `json:"cross-shard-xfer-precompile-epoch,omitempty"`
|
|
|
|
|
|
|
|
// AllowlistEpoch is the first epoch to support allowlist of HIP18
|
|
|
|
AllowlistEpoch *big.Int
|
|
|
|
|
|
|
|
LeaderRotationEpoch *big.Int `json:"leader-rotation-epoch,omitempty"`
|
|
|
|
|
|
|
|
LeaderRotationBlocksCount int `json:"leader-rotation-blocks-count,omitempty"`
|
|
|
|
|
|
|
|
// FeeCollectEpoch is the first epoch that enables txn fees to be collected into the community-managed account.
|
|
|
|
// It should >= StakingEpoch.
|
|
|
|
// Before StakingEpoch, txn fees are paid to miner/leader.
|
|
|
|
// Then before FeeCollectEpoch, txn fees are burned.
|
|
|
|
// After FeeCollectEpoch, txn fees paid to FeeCollector account.
|
|
|
|
FeeCollectEpoch *big.Int
|
|
|
|
}
|
|
|
|
|
|
|
|
// String implements the fmt.Stringer interface.
|
|
|
|
func (c *ChainConfig) String() string {
|
|
|
|
return fmt.Sprintf("{ChainID: %v EthCompatibleChainID: %v EIP155: %v CrossTx: %v Staking: %v CrossLink: %v ReceiptLog: %v SHA3Epoch: %v StakingPrecompileEpoch: %v ChainIdFixEpoch: %v CrossShardXferPrecompileEpoch: %v}",
|
|
|
|
c.ChainID,
|
|
|
|
c.EthCompatibleChainID,
|
|
|
|
c.EIP155Epoch,
|
|
|
|
c.CrossTxEpoch,
|
|
|
|
c.StakingEpoch,
|
|
|
|
c.CrossLinkEpoch,
|
|
|
|
c.ReceiptLogEpoch,
|
|
|
|
c.SHA3Epoch,
|
Resolve harmony-one/bounties#77: Staking precompiles (#3906)
* Resolve harmony-one/bounties#77: Staking precompiles
Create write capable precompiles that can perform staking transactions
Add hard fork logic (EpochTBD) for these precompiles
Tests for new code with at least 80% unit test coverage
Staking library + tests in MaxMustermann2/harmony-staking-precompiles
* Fix small typo in comment
* Run goimports on files to fix Travis
* Do not activate staking precompile on shard 0
* Cascade readOnly to WriteCapableContract
* No overlap in readOnly + writeCapable precompiles
* Use function selector instead of directive
From Solidity, use abi.encodeWithSelector and match it against the
exact ABI of the functions. This allows us to remove the need for
a directive (32) being encoded, and thus saves 28 bytes of data.
* Do not allow contracts to become validators
As discussed with Jacky on #3906
* Merge harmony-one/harmony/main properly this time
* Run goimports
* Update gas calculation for staking precompile
Please see comment in core/vm/contracts_write.go RequiredGas
* Do not allow contract to become validator (2/2)
* Cache StakeMsgs from precompiled transactions
Add the StakeMsgs to ProcessorResult and cascade them in insertChain
* Remove ContractCode fields from validators
Since smart contracts can no longer beecome validators,
this field is superfluous. Remove it from the Wrapper
structure, and do not assign it a value when creating
a validator. Build and goimports checked
* Update comments in response to feedback
(1) Comments to start with function names
(2) Comments for public variables
(3) Comment to match function name RunPrecompiledContract
(4) Clarify that CreateValidatorFunc + EditValidatorFunc are still used
* Fix Travis build by reverting rosetta change
* Add revert capability to 3 staking tx types
- Delegate
- Undelegate
- CollectRewards
* Fix build: Update evm_test for ValidatorWrapper
* Merge main into harmony-staking-precompiles
* Add gas for precompile calls and allow EOA usage
- Each time the precompile is called, charge the base gas fee plus data
cost (if data can be parsed successfully). A gas fee is added to
prevent benevolent contract deployers from subsidizing the staking
transactions for EOAs through repeated assembly `delegatecall`.
- Allow EOAs to use the staking precompile directly. Some changes to
the Solidity library are associated with this change.
- Remove bytes from parsing address, since the ABI unpacks it into an
address format correctly.
- Add or update tests. Test coverage report to be attached to the PR
shortly.
* Run goimports
* Check read only and write capable for overlap
* Handle precompile stakeMsgs for block proposer
The staking precompile generates staking messages which are cascaded to
the block via the EVM in `state_processor.go`. This change cascades them
in `worker.go` to allow block proposers and block verifiers to keep the
same state.
* Run goimports for cf2dfac4081444e36a120c9432f4e..
* Update staking precompile epoch to 2 for localnet
Bring it in line with staking epoch. Change effects all configurations
except mainnet and testnet. `goimports` included.
* Add read only precompile to fetch the epoch num
* Move epoch precompile to 250
* precompiles: left pad the returned epoch number
* chainConfig: check epochs for precompiles
panic if staking precompile epoch < pre staking epoch
* Add staking migration precompile
- Lives at address 251
- Migrates delegations + pending undelegations from address A to B
- Useful if address A is hacked
- Charges gas of 21k + cost of bytes for two addresses
- Does not remove existing delegations, just sets them to zero.
Replicates current undelegate setup
- Unit tests and `goimports` included. Integration test following
shortly in MaxMustermann2/harmony-staking-precompiles
* Migration precompile: merge into staking
Merge the two precompiles into one, add gas calculation for migration
precompile. Move epoch precompile to 251 as a result. When migrating,
add undelegations to `To`'s existing undelegations, if any match the
epoch.
* Add migration gas test, remove panic, add check
In response to review comments, add tests for migration gas wherein
there are 0/1/2 delegations to migrate. Add the index out of bound check
to migration gas calculator and remove panics. Lastly, re-sort
migrated undelegations if no existing undelegation in the same epoch was
found on `To`.
* Move undelegations sorting to end of loop
3 years ago
|
|
|
c.StakingPrecompileEpoch,
|
|
|
|
c.ChainIdFixEpoch,
|
|
|
|
c.CrossShardXferPrecompileEpoch,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
// check if ChainConfig is valid, it will panic if config is invalid. it should only be called in init()
|
|
|
|
func (c *ChainConfig) mustValid() {
|
|
|
|
require := func(cond bool, err string) {
|
|
|
|
if !cond {
|
|
|
|
panic(err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
require(c.FeeCollectEpoch.Cmp(c.StakingEpoch) >= 0,
|
|
|
|
"must satisfy: FeeCollectEpoch >= StakingEpoch")
|
|
|
|
require(c.PreStakingEpoch.Cmp(c.StakingEpoch) < 0,
|
|
|
|
"must satisfy: PreStakingEpoch < StakingEpoch")
|
|
|
|
require(c.StakingPrecompileEpoch.Cmp(c.PreStakingEpoch) >= 0,
|
|
|
|
"must satisfy: StakingPrecompileEpoch >= PreStakingEpoch")
|
|
|
|
require(c.CrossShardXferPrecompileEpoch.Cmp(c.CrossTxEpoch) > 0,
|
|
|
|
"must satisfy: CrossShardXferPrecompileEpoch > CrossTxEpoch")
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsEIP155 returns whether epoch is either equal to the EIP155 fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsEIP155(epoch *big.Int) bool {
|
|
|
|
return isForked(c.EIP155Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// AcceptsCrossTx returns whether cross-shard transaction is accepted in the
|
|
|
|
// given epoch.
|
|
|
|
//
|
|
|
|
// Note that this is different from comparing epoch against CrossTxEpoch.
|
|
|
|
// Cross-shard transaction is accepted from CrossTxEpoch+1 and on, in order to
|
|
|
|
// allow for all shards to roll into CrossTxEpoch and become able to handle
|
|
|
|
// ingress receipts. In other words, cross-shard transaction fields are
|
|
|
|
// introduced and ingress receipts are processed at CrossTxEpoch, but the shard
|
|
|
|
// does not accept cross-shard transactions from clients until CrossTxEpoch+1.
|
|
|
|
func (c *ChainConfig) AcceptsCrossTx(epoch *big.Int) bool {
|
|
|
|
crossTxEpoch := new(big.Int).Add(c.CrossTxEpoch, common.Big1)
|
|
|
|
return isForked(crossTxEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// HasCrossTxFields returns whether blocks in the given epoch includes
|
|
|
|
// cross-shard transaction fields.
|
|
|
|
func (c *ChainConfig) HasCrossTxFields(epoch *big.Int) bool {
|
|
|
|
return isForked(c.CrossTxEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsEthCompatible determines whether it is ethereum compatible epoch
|
|
|
|
func (c *ChainConfig) IsEthCompatible(epoch *big.Int) bool {
|
|
|
|
return isForked(c.EthCompatibleEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsAggregatedRewardEpoch determines whether it is the epoch when rewards are distributed every 64 blocks
|
|
|
|
func (c *ChainConfig) IsAggregatedRewardEpoch(epoch *big.Int) bool {
|
|
|
|
return isForked(c.AggregatedRewardEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsStaking determines whether it is staking epoch
|
|
|
|
func (c *ChainConfig) IsStaking(epoch *big.Int) bool {
|
|
|
|
return isForked(c.StakingEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsSlotsLimited determines whether HIP-16 is enabled
|
|
|
|
func (c *ChainConfig) IsSlotsLimited(epoch *big.Int) bool {
|
|
|
|
return isForked(c.SlotsLimitedEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsFiveSeconds determines whether it is the epoch to change to 5 seconds block time
|
|
|
|
func (c *ChainConfig) IsFiveSeconds(epoch *big.Int) bool {
|
|
|
|
return isForked(c.FiveSecondsEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsTwoSeconds determines whether it is the epoch to change to 3 seconds block time
|
|
|
|
func (c *ChainConfig) IsTwoSeconds(epoch *big.Int) bool {
|
|
|
|
return isForked(c.TwoSecondsEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsSixtyPercent determines whether it is the epoch to reduce internal voting power to 60%
|
|
|
|
func (c *ChainConfig) IsSixtyPercent(epoch *big.Int) bool {
|
|
|
|
return isForked(c.SixtyPercentEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsRedelegation determines whether it is the epoch to support redelegation
|
|
|
|
func (c *ChainConfig) IsRedelegation(epoch *big.Int) bool {
|
|
|
|
return isForked(c.RedelegationEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNoEarlyUnlock determines whether it is the epoch to stop early unlock
|
|
|
|
func (c *ChainConfig) IsNoEarlyUnlock(epoch *big.Int) bool {
|
|
|
|
return isForked(c.NoEarlyUnlockEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsVRF determines whether it is the epoch to enable vrf
|
|
|
|
func (c *ChainConfig) IsVRF(epoch *big.Int) bool {
|
|
|
|
return isForked(c.VRFEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsPrevVRF determines whether it is the epoch to enable previous vrf
|
|
|
|
func (c *ChainConfig) IsPrevVRF(epoch *big.Int) bool {
|
|
|
|
return isForked(c.PrevVRFEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsMinDelegation100 determines whether it is the epoch to reduce min delegation to 100
|
|
|
|
func (c *ChainConfig) IsMinDelegation100(epoch *big.Int) bool {
|
|
|
|
return isForked(c.MinDelegation100Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsMinCommissionRate determines whether it is the epoch to start the policy of 5% min commission
|
|
|
|
func (c *ChainConfig) IsMinCommissionRate(epoch *big.Int) bool {
|
|
|
|
return isForked(c.MinCommissionRateEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsEPoSBound35 determines whether it is the epoch to extend the EPoS bound to 35%
|
|
|
|
func (c *ChainConfig) IsEPoSBound35(epoch *big.Int) bool {
|
|
|
|
return isForked(c.EPoSBound35Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsPreStaking determines whether staking transactions are allowed
|
|
|
|
func (c *ChainConfig) IsPreStaking(epoch *big.Int) bool {
|
|
|
|
return isForked(c.PreStakingEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsQuickUnlock determines whether it's the epoch when the undelegation should be unlocked at end of current epoch
|
|
|
|
func (c *ChainConfig) IsQuickUnlock(epoch *big.Int) bool {
|
|
|
|
return isForked(c.QuickUnlockEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsCrossLink returns whether epoch is either equal to the CrossLink fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsCrossLink(epoch *big.Int) bool {
|
|
|
|
return isForked(c.CrossLinkEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsS3 returns whether epoch is either equal to the S3 fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsS3(epoch *big.Int) bool {
|
|
|
|
return isForked(c.S3Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsDataCopyFixEpoch returns whether epoch has the fix for DataCopy evm bug.
|
|
|
|
func (c *ChainConfig) IsDataCopyFixEpoch(epoch *big.Int) bool {
|
|
|
|
return isForked(c.DataCopyFixEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsIstanbul returns whether epoch is either equal to the Istanbul fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsIstanbul(epoch *big.Int) bool {
|
|
|
|
return isForked(c.IstanbulEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsReceiptLog returns whether epoch is either equal to the ReceiptLog fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsReceiptLog(epoch *big.Int) bool {
|
|
|
|
return isForked(c.ReceiptLogEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsSHA3 returns whether epoch is either equal to the IsSHA3 fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsSHA3(epoch *big.Int) bool {
|
|
|
|
return isForked(c.SHA3Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsHIP6And8Epoch determines whether it is the epoch to support
|
|
|
|
// HIP-6: reduce the internal voting power from 60% to 49%
|
|
|
|
// HIP-8: increase external nodes from 800 to 900
|
|
|
|
func (c *ChainConfig) IsHIP6And8Epoch(epoch *big.Int) bool {
|
|
|
|
return isForked(c.HIP6And8Epoch, epoch)
|
|
|
|
}
|
|
|
|
|
Resolve harmony-one/bounties#77: Staking precompiles (#3906)
* Resolve harmony-one/bounties#77: Staking precompiles
Create write capable precompiles that can perform staking transactions
Add hard fork logic (EpochTBD) for these precompiles
Tests for new code with at least 80% unit test coverage
Staking library + tests in MaxMustermann2/harmony-staking-precompiles
* Fix small typo in comment
* Run goimports on files to fix Travis
* Do not activate staking precompile on shard 0
* Cascade readOnly to WriteCapableContract
* No overlap in readOnly + writeCapable precompiles
* Use function selector instead of directive
From Solidity, use abi.encodeWithSelector and match it against the
exact ABI of the functions. This allows us to remove the need for
a directive (32) being encoded, and thus saves 28 bytes of data.
* Do not allow contracts to become validators
As discussed with Jacky on #3906
* Merge harmony-one/harmony/main properly this time
* Run goimports
* Update gas calculation for staking precompile
Please see comment in core/vm/contracts_write.go RequiredGas
* Do not allow contract to become validator (2/2)
* Cache StakeMsgs from precompiled transactions
Add the StakeMsgs to ProcessorResult and cascade them in insertChain
* Remove ContractCode fields from validators
Since smart contracts can no longer beecome validators,
this field is superfluous. Remove it from the Wrapper
structure, and do not assign it a value when creating
a validator. Build and goimports checked
* Update comments in response to feedback
(1) Comments to start with function names
(2) Comments for public variables
(3) Comment to match function name RunPrecompiledContract
(4) Clarify that CreateValidatorFunc + EditValidatorFunc are still used
* Fix Travis build by reverting rosetta change
* Add revert capability to 3 staking tx types
- Delegate
- Undelegate
- CollectRewards
* Fix build: Update evm_test for ValidatorWrapper
* Merge main into harmony-staking-precompiles
* Add gas for precompile calls and allow EOA usage
- Each time the precompile is called, charge the base gas fee plus data
cost (if data can be parsed successfully). A gas fee is added to
prevent benevolent contract deployers from subsidizing the staking
transactions for EOAs through repeated assembly `delegatecall`.
- Allow EOAs to use the staking precompile directly. Some changes to
the Solidity library are associated with this change.
- Remove bytes from parsing address, since the ABI unpacks it into an
address format correctly.
- Add or update tests. Test coverage report to be attached to the PR
shortly.
* Run goimports
* Check read only and write capable for overlap
* Handle precompile stakeMsgs for block proposer
The staking precompile generates staking messages which are cascaded to
the block via the EVM in `state_processor.go`. This change cascades them
in `worker.go` to allow block proposers and block verifiers to keep the
same state.
* Run goimports for cf2dfac4081444e36a120c9432f4e..
* Update staking precompile epoch to 2 for localnet
Bring it in line with staking epoch. Change effects all configurations
except mainnet and testnet. `goimports` included.
* Add read only precompile to fetch the epoch num
* Move epoch precompile to 250
* precompiles: left pad the returned epoch number
* chainConfig: check epochs for precompiles
panic if staking precompile epoch < pre staking epoch
* Add staking migration precompile
- Lives at address 251
- Migrates delegations + pending undelegations from address A to B
- Useful if address A is hacked
- Charges gas of 21k + cost of bytes for two addresses
- Does not remove existing delegations, just sets them to zero.
Replicates current undelegate setup
- Unit tests and `goimports` included. Integration test following
shortly in MaxMustermann2/harmony-staking-precompiles
* Migration precompile: merge into staking
Merge the two precompiles into one, add gas calculation for migration
precompile. Move epoch precompile to 251 as a result. When migrating,
add undelegations to `To`'s existing undelegations, if any match the
epoch.
* Add migration gas test, remove panic, add check
In response to review comments, add tests for migration gas wherein
there are 0/1/2 delegations to migrate. Add the index out of bound check
to migration gas calculator and remove panics. Lastly, re-sort
migrated undelegations if no existing undelegation in the same epoch was
found on `To`.
* Move undelegations sorting to end of loop
3 years ago
|
|
|
// IsStakingPrecompileEpoch determines whether staking
|
|
|
|
// precompiles are available in the EVM
|
|
|
|
func (c *ChainConfig) IsStakingPrecompile(epoch *big.Int) bool {
|
|
|
|
return isForked(c.StakingPrecompileEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsCrossShardXferPrecompile determines whether the
|
|
|
|
// Cross Shard Transfer Precompile is available in the EVM
|
|
|
|
func (c *ChainConfig) IsCrossShardXferPrecompile(epoch *big.Int) bool {
|
|
|
|
return isForked(c.CrossShardXferPrecompileEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsChainIdFix returns whether epoch is either equal to the ChainId Fix fork epoch or greater.
|
|
|
|
func (c *ChainConfig) IsChainIdFix(epoch *big.Int) bool {
|
|
|
|
return isForked(c.ChainIdFixEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsAllowlistEpoch determines whether IsAllowlist of HIP18 is enabled
|
|
|
|
func (c *ChainConfig) IsAllowlistEpoch(epoch *big.Int) bool {
|
|
|
|
return isForked(c.AllowlistEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *ChainConfig) IsLeaderRotation(epoch *big.Int) bool {
|
|
|
|
return isForked(c.LeaderRotationEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsFeeCollectEpoch determines whether Txn Fees will be collected into the community-managed account.
|
|
|
|
func (c *ChainConfig) IsFeeCollectEpoch(epoch *big.Int) bool {
|
|
|
|
return isForked(c.FeeCollectEpoch, epoch)
|
|
|
|
}
|
|
|
|
|
|
|
|
// UpdateEthChainIDByShard update the ethChainID based on shard ID.
|
|
|
|
func UpdateEthChainIDByShard(shardID uint32) {
|
|
|
|
once.Do(func() {
|
|
|
|
MainnetChainConfig.EthCompatibleChainID = big.NewInt(0).Add(MainnetChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
TestnetChainConfig.EthCompatibleChainID = big.NewInt(0).Add(TestnetChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
PangaeaChainConfig.EthCompatibleChainID = big.NewInt(0).Add(PangaeaChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
PartnerChainConfig.EthCompatibleChainID = big.NewInt(0).Add(PartnerChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
StressnetChainConfig.EthCompatibleChainID = big.NewInt(0).Add(StressnetChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
LocalnetChainConfig.EthCompatibleChainID = big.NewInt(0).Add(LocalnetChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
AllProtocolChanges.EthCompatibleChainID = big.NewInt(0).Add(AllProtocolChanges.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
TestChainConfig.EthCompatibleChainID = big.NewInt(0).Add(TestChainConfig.EthCompatibleChainID, big.NewInt(int64(shardID)))
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsEthCompatible returns whether the chainID is for ethereum compatible txn or not
|
|
|
|
func IsEthCompatible(chainID *big.Int) bool {
|
|
|
|
return chainID.Cmp(EthMainnetShard0ChainID) >= 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// GasTable returns the gas table corresponding to the current phase (homestead or homestead reprice).
|
|
|
|
//
|
|
|
|
// The returned GasTable's fields shouldn't, under any circumstances, be changed.
|
|
|
|
func (c *ChainConfig) GasTable(epoch *big.Int) GasTable {
|
|
|
|
if epoch == nil {
|
|
|
|
return GasTableR3
|
|
|
|
}
|
|
|
|
switch {
|
|
|
|
case c.IsS3(epoch):
|
|
|
|
return GasTableS3
|
|
|
|
default:
|
|
|
|
return GasTableR3
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// isForked returns whether a fork scheduled at epoch s is active at the given head epoch.
|
|
|
|
func isForked(s, epoch *big.Int) bool {
|
|
|
|
if s == nil || epoch == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return s.Cmp(epoch) <= 0
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rules wraps ChainConfig and is merely syntactic sugar or can be used for functions
|
|
|
|
// that do not have or require information about the block.
|
|
|
|
//
|
|
|
|
// Rules is a one time interface meaning that it shouldn't be used in between transition
|
|
|
|
// phases.
|
|
|
|
type Rules struct {
|
|
|
|
ChainID *big.Int
|
|
|
|
EthChainID *big.Int
|
|
|
|
// gas
|
|
|
|
IsS3,
|
|
|
|
// precompiles
|
|
|
|
IsIstanbul, IsVRF, IsPrevVRF, IsSHA3,
|
|
|
|
IsStakingPrecompile, IsCrossShardXferPrecompile,
|
|
|
|
// eip-155 chain id fix
|
|
|
|
IsChainIdFix bool
|
|
|
|
}
|
|
|
|
|
|
|
|
// Rules ensures c's ChainID is not nil.
|
|
|
|
// The Rules object is only used by The EVM
|
|
|
|
func (c *ChainConfig) Rules(epoch *big.Int) Rules {
|
|
|
|
chainID := c.ChainID
|
|
|
|
if chainID == nil {
|
|
|
|
chainID = new(big.Int)
|
|
|
|
}
|
|
|
|
ethChainID := c.EthCompatibleChainID
|
|
|
|
if ethChainID == nil {
|
|
|
|
ethChainID = new(big.Int)
|
|
|
|
}
|
|
|
|
return Rules{
|
|
|
|
ChainID: new(big.Int).Set(chainID),
|
|
|
|
EthChainID: new(big.Int).Set(ethChainID),
|
|
|
|
IsS3: c.IsS3(epoch),
|
|
|
|
IsIstanbul: c.IsIstanbul(epoch),
|
|
|
|
IsVRF: c.IsVRF(epoch),
|
|
|
|
IsPrevVRF: c.IsPrevVRF(epoch),
|
|
|
|
IsSHA3: c.IsSHA3(epoch),
|
|
|
|
IsStakingPrecompile: c.IsStakingPrecompile(epoch),
|
|
|
|
IsCrossShardXferPrecompile: c.IsCrossShardXferPrecompile(epoch),
|
|
|
|
IsChainIdFix: c.IsChainIdFix(epoch),
|
|
|
|
}
|
|
|
|
}
|