-
Notifications
You must be signed in to change notification settings - Fork 2
Expand file tree
/
Copy pathProxy.sol
More file actions
104 lines (94 loc) · 3.58 KB
/
Proxy.sol
File metadata and controls
104 lines (94 loc) · 3.58 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
// SPDX-License-Identifier: MIT
pragma solidity 0.6.11;
import "../Dependencies/Ownable.sol";
/**
* @title Base Proxy contract.
*
* Adapted version of https://github.com/DistributedCollective/Sovryn-smart-contracts/blob/development/contracts/proxy/Proxy.sol
*
* @notice The proxy performs delegated calls to the contract implementation
* it is pointing to. This way upgradable contracts are possible on blockchain.
*
* Delegating proxy contracts are widely used for both upgradeability and gas
* savings. These proxies rely on a logic contract (also known as implementation
* contract or master copy) that is called using delegatecall. This allows
* proxies to keep a persistent state (storage and balance) while the code is
* delegated to the logic contract.
*
* Proxy contract is meant to be inherited and its internal functions
* _setImplementation and _setOwner to be called when upgrades become
* neccessary.
*
* The loan token (iToken) contract as well as the protocol contract act as
* proxies, delegating all calls to underlying contracts. Therefore, if you
* want to interact with them using web3, you need to use the ABIs from the
* contracts containing the actual logic or the interface contract.
* ABI for LoanToken contracts: LoanTokenLogicStandard
* ABI for Protocol contract: ISovryn
*
* @dev UpgradableProxy is the contract that inherits Proxy and wraps these
* functions.
* */
contract Proxy is Ownable {
bytes32 private constant KEY_IMPLEMENTATION = keccak256("key.implementation");
event ImplementationChanged(
address indexed _oldImplementation,
address indexed _newImplementation
);
/**
* @notice Set address of the implementation.
* @param _implementation Address of the implementation.
* */
function _setImplementation(address _implementation) internal {
require(_implementation != address(0), "Proxy::setImplementation: invalid address");
emit ImplementationChanged(getImplementation(), _implementation);
bytes32 key = KEY_IMPLEMENTATION;
assembly {
sstore(key, _implementation)
}
}
/**
* @notice Return address of the implementation.
* @return _implementation Address of the implementation.
* */
function getImplementation() public view returns (address _implementation) {
bytes32 key = KEY_IMPLEMENTATION;
assembly {
_implementation := sload(key)
}
}
/**
* @notice Fallback function performs a delegate call
* to the actual implementation address is pointing this proxy.
* Returns whatever the implementation call returns.
* */
fallback() external payable {
delegate();
}
/**
* @notice Fallback function performs a delegate call
* to the actual implementation address is pointing this proxy.
* Returns whatever the implementation call returns.
* */
receive() external payable {
delegate();
}
function delegate() internal {
address implementation = getImplementation();
require(implementation != address(0), "Proxy::(): implementation not found");
assembly {
let pointer := mload(0x40)
calldatacopy(pointer, 0, calldatasize())
let result := delegatecall(gas(), implementation, pointer, calldatasize(), 0, 0)
let size := returndatasize()
returndatacopy(pointer, 0, size)
switch result
case 0 {
revert(pointer, size)
}
default {
return(pointer, size)
}
}
}
}