forked from ethereum-optimism/optimism
-
Notifications
You must be signed in to change notification settings - Fork 5
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
Showing
3 changed files
with
211 additions
and
25 deletions.
There are no files selected for viewing
Submodule solady-v0.0.245
added at
e0ef35
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,161 @@ | ||
pragma solidity ^0.5.13; | ||
/* solhint-disable no-inline-assembly, no-complex-fallback, avoid-low-level-calls */ | ||
|
||
import "openzeppelin-solidity/contracts/utils/Address.sol"; | ||
|
||
/** | ||
* @title A Proxy utilizing the Unstructured Storage pattern. | ||
*/ | ||
contract CeloProxy { | ||
// Used to store the address of the owner. | ||
bytes32 private constant OWNER_POSITION = bytes32(uint256(keccak256("eip1967.proxy.admin")) - 1); | ||
// Used to store the address of the implementation contract. | ||
bytes32 private constant IMPLEMENTATION_POSITION = | ||
bytes32(uint256(keccak256("eip1967.proxy.implementation")) - 1); | ||
|
||
event OwnerSet(address indexed owner); | ||
event ImplementationSet(address indexed implementation); | ||
|
||
/** | ||
* @notice Throws if called by any account other than the owner. | ||
*/ | ||
modifier onlyOwner() { | ||
require(msg.sender == _getOwner(), "sender was not owner"); | ||
_; | ||
} | ||
|
||
constructor() public { | ||
_setOwner(msg.sender); | ||
} | ||
|
||
/** | ||
* @notice Delegates calls to the implementation contract. | ||
*/ | ||
function() external payable { | ||
bytes32 implementationPosition = IMPLEMENTATION_POSITION; | ||
|
||
address implementationAddress; | ||
|
||
// Load the address of the implementation contract from an explicit storage slot. | ||
assembly { | ||
implementationAddress := sload(implementationPosition) | ||
} | ||
|
||
// Avoid checking if address is a contract or executing delegated call when | ||
// implementation address is 0x0 | ||
require(implementationAddress != address(0), "No Implementation set"); | ||
require(Address.isContract(implementationAddress), "Invalid contract address"); | ||
|
||
assembly { | ||
// Extract the position of the transaction data (i.e. function ID and arguments). | ||
let newCallDataPosition := mload(0x40) | ||
mstore(0x40, add(newCallDataPosition, calldatasize)) | ||
calldatacopy(newCallDataPosition, 0, calldatasize) | ||
|
||
// Call the smart contract at `implementationAddress` in the context of the proxy contract, | ||
// with the same msg.sender and value. | ||
let delegatecallSuccess := delegatecall( | ||
gas, | ||
implementationAddress, | ||
newCallDataPosition, | ||
calldatasize, | ||
0, | ||
0 | ||
) | ||
|
||
// Copy the return value of the call so it can be returned. | ||
let returnDataSize := returndatasize | ||
let returnDataPosition := mload(0x40) | ||
mstore(0x40, add(returnDataPosition, returnDataSize)) | ||
returndatacopy(returnDataPosition, 0, returnDataSize) | ||
|
||
// Revert or return depending on whether or not the call was successful. | ||
switch delegatecallSuccess | ||
case 0 { | ||
revert(returnDataPosition, returnDataSize) | ||
} | ||
default { | ||
return(returnDataPosition, returnDataSize) | ||
} | ||
} | ||
} | ||
|
||
/** | ||
* @notice Transfers ownership of Proxy to a new owner. | ||
* @param newOwner Address of the new owner account. | ||
*/ | ||
function _transferOwnership(address newOwner) external onlyOwner { | ||
_setOwner(newOwner); | ||
} | ||
|
||
/** | ||
* @notice Sets the address of the implementation contract and calls into it. | ||
* @param implementation Address of the new target contract. | ||
* @param callbackData The abi-encoded function call to perform in the implementation | ||
* contract. | ||
* @dev Throws if the initialization callback fails. | ||
* @dev If the target contract does not need initialization, use | ||
* setImplementation instead. | ||
*/ | ||
function _setAndInitializeImplementation( | ||
address implementation, | ||
bytes calldata callbackData | ||
) external payable onlyOwner { | ||
_setImplementation(implementation); | ||
bool success; | ||
bytes memory returnValue; | ||
(success, returnValue) = implementation.delegatecall(callbackData); | ||
require(success, "initialization callback failed"); | ||
} | ||
|
||
/** | ||
* @notice Returns the implementation address. | ||
*/ | ||
function _getImplementation() external view returns (address implementation) { | ||
bytes32 implementationPosition = IMPLEMENTATION_POSITION; | ||
// Load the address of the implementation contract from an explicit storage slot. | ||
assembly { | ||
implementation := sload(implementationPosition) | ||
} | ||
} | ||
|
||
/** | ||
* @notice Sets the address of the implementation contract. | ||
* @param implementation Address of the new target contract. | ||
* @dev If the target contract needs to be initialized, call | ||
* setAndInitializeImplementation instead. | ||
*/ | ||
function _setImplementation(address implementation) public onlyOwner { | ||
bytes32 implementationPosition = IMPLEMENTATION_POSITION; | ||
|
||
require(Address.isContract(implementation), "Invalid contract address"); | ||
|
||
// Store the address of the implementation contract in an explicit storage slot. | ||
assembly { | ||
sstore(implementationPosition, implementation) | ||
} | ||
|
||
emit ImplementationSet(implementation); | ||
} | ||
|
||
/** | ||
* @notice Returns the Proxy owner's address. | ||
*/ | ||
function _getOwner() public view returns (address owner) { | ||
bytes32 position = OWNER_POSITION; | ||
// Load the address of the contract owner from an explicit storage slot. | ||
assembly { | ||
owner := sload(position) | ||
} | ||
} | ||
|
||
function _setOwner(address newOwner) internal { | ||
require(newOwner != address(0), "owner cannot be 0"); | ||
bytes32 position = OWNER_POSITION; | ||
// Store the address of the contract owner in an explicit storage slot. | ||
assembly { | ||
sstore(position, newOwner) | ||
} | ||
emit OwnerSet(newOwner); | ||
} | ||
} |