-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathERC4626LinearPool.sol
94 lines (80 loc) · 3.73 KB
/
ERC4626LinearPool.sol
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
// SPDX-License-Identifier: GPL-3.0-or-later
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
pragma solidity ^0.7.0;
pragma experimental ABIEncoderV2;
import "@balancer-labs/v2-interfaces/contracts/solidity-utils/misc/IERC4626.sol";
import "@balancer-labs/v2-solidity-utils/contracts/openzeppelin/ERC20.sol";
import "@balancer-labs/v2-solidity-utils/contracts/math/Math.sol";
import "../LinearPool.sol";
contract ERC4626LinearPool is LinearPool {
using Math for uint256;
uint256 private immutable _rateScaleFactor;
constructor(
IVault vault,
string memory name,
string memory symbol,
IERC20 mainToken,
IERC4626 wrappedToken,
uint256 upperTarget,
uint256 swapFeePercentage,
uint256 pauseWindowDuration,
uint256 bufferPeriodDuration,
address owner
)
LinearPool(
vault,
name,
symbol,
mainToken,
wrappedToken,
upperTarget,
new address[](2),
swapFeePercentage,
pauseWindowDuration,
bufferPeriodDuration,
owner
)
{
// We do NOT enforce mainToken == wrappedToken.asset() even
// though this is the expected behavior in most cases. Instead,
// we assume a 1:1 relationship between mainToken and
// wrappedToken.asset(), but they do not have to be the same
// token. It is vitally important that this 1:1 relationship is
// respected, or the pool will not function as intended.
//
// This allows for use cases where the wrappedToken is
// double-wrapped into an ERC-4626 token. For example, consider
// a linear pool whose goal is to pair DAI with aDAI. Because
// aDAI is a rebasing token, it needs to be wrapped, and let's
// say an ERC-4626 wrapper is chosen for compatibility with this
// linear pool. Then wrappedToken.asset() will return aDAI,
// whereas mainToken is DAI. But the 1:1 relationship holds, and
// the pool is still valid.
// _getWrappedTokenRate is scaled e18, so we may need to scale IERC4626.convertToAssets()
uint256 wrappedTokenDecimals = ERC20(address(wrappedToken)).decimals();
uint256 mainTokenDecimals = ERC20(address(mainToken)).decimals();
// This is always positive because we only accept tokens with <= 18 decimals
uint256 digitsDifference = Math.add(18, wrappedTokenDecimals).sub(mainTokenDecimals);
_rateScaleFactor = 10**digitsDifference;
}
function _getWrappedTokenRate() internal view override returns (uint256) {
IERC4626 wrappedToken = IERC4626(address(getWrappedToken()));
// Main tokens per 1e18 wrapped token wei
// decimals: main + (18 - wrapped)
uint256 assetsPerShare = wrappedToken.convertToAssets(FixedPoint.ONE);
// This function returns a 18 decimal fixed point number
// assetsPerShare decimals: 18 + main - wrapped
// _rateScaleFactor decimals: 18 - main + wrapped
uint256 rate = assetsPerShare.mul(_rateScaleFactor).divDown(FixedPoint.ONE);
return rate;
}
}