mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-31 23:41:22 +00:00
533e12b6e7
Former-commit-id: e8aa99e8cae2f59728986c9bf079ad399bb1d8ed [formerly 7e7f88907c44c4a7cbefd2d0095dee33a86265e8] Former-commit-id: 7e8ef4440f41760cf346378916242888b44f609f
254 lines
8.5 KiB
Solidity
254 lines
8.5 KiB
Solidity
pragma solidity ^0.4.19;
|
|
|
|
contract SMC {
|
|
event TxToShard(address indexed to, int indexed shardId, int receiptId);
|
|
event CollationAdded(int indexed shardId, uint expectedPeriodNumber,
|
|
bytes32 periodStartPrevHash, bytes32 parentHash,
|
|
bytes32 transactionRoot, address coinbase,
|
|
bytes32 stateRoot, bytes32 receiptRoot,
|
|
int number, bool isNewHead, int score);
|
|
event Deposit(address collator, int index);
|
|
event Withdraw(int index);
|
|
|
|
struct Collator {
|
|
// Amount of wei the collator holds
|
|
uint deposit;
|
|
// The collator's address
|
|
address addr;
|
|
}
|
|
|
|
struct CollationHeader {
|
|
bytes32 parentHash;
|
|
int score;
|
|
}
|
|
|
|
struct Receipt {
|
|
int shardId;
|
|
uint txStartgas;
|
|
uint txGasprice;
|
|
uint value;
|
|
bytes32 data;
|
|
address sender;
|
|
address to;
|
|
}
|
|
|
|
// Packed variables to be used in addHeader
|
|
struct HeaderVars {
|
|
bytes32 entireHeaderHash;
|
|
int score;
|
|
address collatorAddr;
|
|
bool isNewHead;
|
|
}
|
|
|
|
// collatorId => Collators
|
|
mapping (int => Collator) public collators;
|
|
// shardId => (headerHash => CollationHeader)
|
|
mapping (int => mapping (bytes32 => CollationHeader)) public collationHeaders;
|
|
// receiptId => Receipt
|
|
mapping (int => Receipt) public receipts;
|
|
// shardId => headerHash
|
|
mapping (int => bytes32) shardHead;
|
|
|
|
// Number of collators
|
|
int public numCollators;
|
|
// Number of receipts
|
|
int numReceipts;
|
|
// Indexs of empty slots caused by the function `withdraw`
|
|
mapping (int => int) emptySlotsStack;
|
|
// The top index of the stack in empty_slots_stack
|
|
int emptySlotsStackTop;
|
|
// Has the collator deposited before?
|
|
mapping (address => bool) public isCollatorDeposited;
|
|
|
|
// Constant values
|
|
uint constant periodLength = 5;
|
|
int constant public shardCount = 100;
|
|
// The exact deposit size which you have to deposit to become a collator
|
|
uint constant depositSize = 100 ether;
|
|
// Number of periods ahead of current period, which the contract
|
|
// is able to return the collator of that period
|
|
uint constant lookAheadPeriods = 4;
|
|
|
|
// Log the latest period number of the shard
|
|
mapping (int => int) public periodHead;
|
|
|
|
function SMC() public {
|
|
}
|
|
|
|
// Returns the gas limit that collations can currently have (by default make
|
|
// this function always answer 10 million).
|
|
function getCollationGasLimit() public pure returns(uint) {
|
|
return 10000000;
|
|
}
|
|
|
|
// Uses a block hash as a seed to pseudorandomly select a signer from the collator pool.
|
|
// [TODO] Chance of being selected should be proportional to the collator's deposit.
|
|
// Should be able to return a value for the current period or any future period up to.
|
|
function getEligibleProposer(int _shardId, uint _period) public view returns(address) {
|
|
require(_period >= lookAheadPeriods);
|
|
require((_period - lookAheadPeriods) * periodLength < block.number);
|
|
require(numCollators > 0);
|
|
// [TODO] Should check further if this safe or not
|
|
return collators[
|
|
int(
|
|
uint(
|
|
keccak256(
|
|
uint(block.blockhash((_period - lookAheadPeriods) * periodLength)),
|
|
_shardId
|
|
)
|
|
) %
|
|
uint(getCollatorsMaxIndex())
|
|
)
|
|
].addr;
|
|
}
|
|
|
|
function deposit() public payable returns(int) {
|
|
require(!isCollatorDeposited[msg.sender]);
|
|
require(msg.value == depositSize);
|
|
// Find the empty slot index in collators pool
|
|
int index;
|
|
if (!isStackEmpty())
|
|
index = stackPop();
|
|
else
|
|
index = int(numCollators);
|
|
|
|
collators[index] = Collator({
|
|
deposit: msg.value,
|
|
addr: msg.sender
|
|
});
|
|
++numCollators;
|
|
isCollatorDeposited[msg.sender] = true;
|
|
|
|
Deposit(msg.sender, index);
|
|
return index;
|
|
}
|
|
|
|
// Removes the collator from the collator pool and refunds the deposited ether
|
|
function withdraw(int _collatorIndex) public {
|
|
require(msg.sender == collators[_collatorIndex].addr);
|
|
// [FIXME] Should consider calling the collator's contract, might be useful
|
|
// when the collator is a contract.
|
|
collators[_collatorIndex].addr.transfer(collators[_collatorIndex].deposit);
|
|
isCollatorDeposited[collators[_collatorIndex].addr] = false;
|
|
delete collators[_collatorIndex];
|
|
stackPush(_collatorIndex);
|
|
--numCollators;
|
|
Withdraw(_collatorIndex);
|
|
}
|
|
|
|
// Attempts to process a collation header, returns true on success, reverts on failure.
|
|
function addHeader(int _shardId, uint _expectedPeriodNumber, bytes32 _periodStartPrevHash,
|
|
bytes32 _parentHash, bytes32 _transactionRoot,
|
|
address _coinbase, bytes32 _stateRoot, bytes32 _receiptRoot,
|
|
int _number) public returns(bool) {
|
|
HeaderVars memory headerVars;
|
|
|
|
// Check if the header is valid
|
|
require((_shardId >= 0) && (_shardId < shardCount));
|
|
require(block.number >= periodLength);
|
|
require(_expectedPeriodNumber == block.number / periodLength);
|
|
require(_periodStartPrevHash == block.blockhash(_expectedPeriodNumber * periodLength - 1));
|
|
|
|
// Check if this header already exists
|
|
headerVars.entireHeaderHash = keccak256(_shardId, _expectedPeriodNumber, _periodStartPrevHash,
|
|
_parentHash, _transactionRoot, bytes32(_coinbase),
|
|
_stateRoot, _receiptRoot, _number);
|
|
assert(collationHeaders[_shardId][headerVars.entireHeaderHash].score == 0);
|
|
// Check whether the parent exists.
|
|
// if (parent_collation_hash == 0), i.e., is the genesis,
|
|
// then there is no need to check.
|
|
if (_parentHash != 0x0)
|
|
assert(collationHeaders[_shardId][_parentHash].score > 0);
|
|
// Check if only one collation in one period
|
|
assert(periodHead[_shardId] < int(_expectedPeriodNumber));
|
|
|
|
// Check the signature with validation_code_addr
|
|
headerVars.collatorAddr = getEligibleProposer(_shardId, block.number/periodLength);
|
|
require(headerVars.collatorAddr != 0x0);
|
|
require(msg.sender == headerVars.collatorAddr);
|
|
|
|
// Check score == collationNumber
|
|
headerVars.score = collationHeaders[_shardId][_parentHash].score + 1;
|
|
require(_number == headerVars.score);
|
|
|
|
// Add the header
|
|
collationHeaders[_shardId][headerVars.entireHeaderHash] = CollationHeader({
|
|
parentHash: _parentHash,
|
|
score: headerVars.score
|
|
});
|
|
|
|
// Update the latest period number
|
|
periodHead[_shardId] = int(_expectedPeriodNumber);
|
|
|
|
// Determine the head
|
|
if (headerVars.score > collationHeaders[_shardId][shardHead[_shardId]].score) {
|
|
shardHead[_shardId] = headerVars.entireHeaderHash;
|
|
headerVars.isNewHead = true;
|
|
}
|
|
|
|
CollationAdded(_shardId, _expectedPeriodNumber, _periodStartPrevHash,
|
|
_parentHash, _transactionRoot, _coinbase, _stateRoot,
|
|
_receiptRoot, _number, headerVars.isNewHead, headerVars.score);
|
|
|
|
return true;
|
|
}
|
|
|
|
// Records a request to deposit msg.value ETH to address to in shard shard_id
|
|
// during a future collation. Saves a `receipt ID` for this request,
|
|
// also saving `msg.sender`, `msg.value`, `to`, `shard_id`, `startgas`,
|
|
// `gasprice`, and `data`.
|
|
function txToShard(address _to, int _shardId, uint _txStartgas, uint _txGasprice,
|
|
bytes12 _data) public payable returns(int) {
|
|
receipts[numReceipts] = Receipt({
|
|
shardId: _shardId,
|
|
txStartgas: _txStartgas,
|
|
txGasprice: _txGasprice,
|
|
value: msg.value,
|
|
sender: msg.sender,
|
|
to: _to,
|
|
data: _data
|
|
});
|
|
var receiptId = numReceipts;
|
|
++numReceipts;
|
|
|
|
TxToShard(_to, _shardId, receiptId);
|
|
return receiptId;
|
|
}
|
|
|
|
function updateGasPrice(int _receiptId, uint _txGasprice) public payable returns(bool) {
|
|
require(receipts[_receiptId].sender == msg.sender);
|
|
receipts[_receiptId].txGasprice = _txGasprice;
|
|
return true;
|
|
}
|
|
|
|
function isStackEmpty() internal view returns(bool) {
|
|
return emptySlotsStackTop == 0;
|
|
}
|
|
|
|
function stackPush(int index) internal {
|
|
emptySlotsStack[emptySlotsStackTop] = index;
|
|
++emptySlotsStackTop;
|
|
}
|
|
|
|
function stackPop() internal returns(int) {
|
|
if (isStackEmpty())
|
|
return -1;
|
|
--emptySlotsStackTop;
|
|
return emptySlotsStack[emptySlotsStackTop];
|
|
}
|
|
|
|
function getCollatorsMaxIndex() internal view returns(int) {
|
|
int activateCollatorNum = 0;
|
|
int allCollatorSlotsNum = numCollators + emptySlotsStackTop;
|
|
|
|
// TODO: any better way to iterate the mapping?
|
|
for (int i = 0; i < 1024; ++i) {
|
|
if (i >= allCollatorSlotsNum)
|
|
break;
|
|
if (collators[i].addr != 0x0)
|
|
activateCollatorNum += 1;
|
|
}
|
|
return activateCollatorNum + emptySlotsStackTop;
|
|
}
|
|
}
|