2020-12-11 22:52:00 +08:00
|
|
|
|
/*
|
|
|
|
|
|
|
|
|
|
|
|
Copyright 2020 DODO ZOO.
|
|
|
|
|
|
SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
|
|
|
|
pragma solidity 0.6.9;
|
|
|
|
|
|
pragma experimental ABIEncoderV2;
|
|
|
|
|
|
|
|
|
|
|
|
import {SafeMath} from "../../lib/SafeMath.sol";
|
|
|
|
|
|
import {SafeERC20} from "../../lib/SafeERC20.sol";
|
|
|
|
|
|
import {DecimalMath} from "../../lib/DecimalMath.sol";
|
|
|
|
|
|
import {IERC20} from "../../intf/IERC20.sol";
|
|
|
|
|
|
import {IDVM} from "../../DODOVendingMachine/intf/IDVM.sol";
|
2020-12-12 15:53:42 +08:00
|
|
|
|
import {IUnownedDVMFactory} from "../../Factory/UnownedDVMFactory.sol";
|
|
|
|
|
|
import {CPStorage} from "./CPStorage.sol";
|
2020-12-11 22:52:00 +08:00
|
|
|
|
import {PMMPricing} from "../../lib/PMMPricing.sol";
|
|
|
|
|
|
|
2020-12-12 15:53:42 +08:00
|
|
|
|
contract CPFunding is CPStorage {
|
2020-12-11 22:52:00 +08:00
|
|
|
|
using SafeERC20 for IERC20;
|
|
|
|
|
|
|
|
|
|
|
|
// ============ BID & CALM PHASE ============
|
|
|
|
|
|
|
|
|
|
|
|
modifier isBidderAllow(address bidder) {
|
|
|
|
|
|
require(_BIDDER_PERMISSION_.isAllowed(bidder), "BIDDER_NOT_ALLOWED");
|
|
|
|
|
|
_;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function bid(address to) external phaseBid preventReentrant isBidderAllow(to) {
|
|
|
|
|
|
uint256 input = _getQuoteInput();
|
|
|
|
|
|
uint256 mtFee = DecimalMath.mulFloor(input, _MT_FEE_RATE_MODEL_.getFeeRate(to));
|
|
|
|
|
|
_transferQuoteOut(_MAINTAINER_, mtFee);
|
|
|
|
|
|
_mintShares(to, input.sub(mtFee));
|
|
|
|
|
|
_sync();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function cancel(address assetTo, uint256 amount) external phaseBidOrCalm preventReentrant {
|
|
|
|
|
|
require(_SHARES_[msg.sender] >= amount, "SHARES_NOT_ENOUGH");
|
|
|
|
|
|
_burnShares(msg.sender, amount);
|
|
|
|
|
|
_transferQuoteOut(assetTo, amount);
|
|
|
|
|
|
_sync();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function _mintShares(address to, uint256 amount) internal {
|
|
|
|
|
|
_SHARES_[to] = _SHARES_[to].add(amount);
|
|
|
|
|
|
_TOTAL_SHARES_ = _TOTAL_SHARES_.add(amount);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function _burnShares(address from, uint256 amount) internal {
|
|
|
|
|
|
_SHARES_[from] = _SHARES_[from].sub(amount);
|
|
|
|
|
|
_TOTAL_SHARES_ = _TOTAL_SHARES_.sub(amount);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ============ SETTLEMENT ============
|
|
|
|
|
|
|
|
|
|
|
|
function settle() external phaseSettlement preventReentrant {
|
2020-12-13 17:47:47 +08:00
|
|
|
|
_settle();
|
2020-12-11 22:52:00 +08:00
|
|
|
|
|
|
|
|
|
|
(uint256 poolBase, uint256 poolQuote, uint256 ownerQuote) = getSettleResult();
|
|
|
|
|
|
_UNUSED_QUOTE_ = _QUOTE_TOKEN_.balanceOf(address(this)).sub(poolQuote).sub(ownerQuote);
|
|
|
|
|
|
_UNUSED_BASE_ = _BASE_TOKEN_.balanceOf(address(this)).sub(poolBase);
|
|
|
|
|
|
|
|
|
|
|
|
// 这里的目的是让开盘价尽量等于avgPrice
|
|
|
|
|
|
// 我们统一设定k=1,如果quote和base不平衡,就必然要截断一边
|
|
|
|
|
|
// DVM截断了quote,所以如果进入池子的quote很多,就要把quote设置成DVM的base
|
|
|
|
|
|
// m = avgPrice
|
|
|
|
|
|
// i = m (1-quote/(m*base))
|
|
|
|
|
|
// if quote = m*base i = 1
|
|
|
|
|
|
// if quote > m*base reverse
|
2020-12-12 15:53:42 +08:00
|
|
|
|
{
|
|
|
|
|
|
uint256 avgPrice = DecimalMath.divCeil(poolQuote.add(ownerQuote), _UNUSED_BASE_);
|
|
|
|
|
|
uint256 baseDepth = DecimalMath.mulFloor(avgPrice, poolBase);
|
|
|
|
|
|
address _poolBaseToken;
|
|
|
|
|
|
address _poolQuoteToken;
|
|
|
|
|
|
uint256 _poolI;
|
|
|
|
|
|
if (poolQuote == baseDepth) {
|
|
|
|
|
|
_poolBaseToken = address(_BASE_TOKEN_);
|
|
|
|
|
|
_poolQuoteToken = address(_QUOTE_TOKEN_);
|
|
|
|
|
|
_poolI = 1;
|
|
|
|
|
|
} else if (poolQuote < baseDepth) {
|
2020-12-13 17:47:47 +08:00
|
|
|
|
// poolI up round
|
2020-12-12 15:53:42 +08:00
|
|
|
|
_poolBaseToken = address(_BASE_TOKEN_);
|
|
|
|
|
|
_poolQuoteToken = address(_QUOTE_TOKEN_);
|
2020-12-13 17:47:47 +08:00
|
|
|
|
uint256 ratio = DecimalMath.ONE.sub(DecimalMath.divCeil(poolQuote, baseDepth));
|
2020-12-12 15:53:42 +08:00
|
|
|
|
_poolI = avgPrice.mul(ratio).mul(ratio).divCeil(DecimalMath.ONE2);
|
|
|
|
|
|
} else if (poolQuote > baseDepth) {
|
2020-12-13 17:47:47 +08:00
|
|
|
|
// poolI down round
|
2020-12-12 15:53:42 +08:00
|
|
|
|
_poolBaseToken = address(_QUOTE_TOKEN_);
|
|
|
|
|
|
_poolQuoteToken = address(_BASE_TOKEN_);
|
|
|
|
|
|
uint256 ratio = DecimalMath.ONE.sub(DecimalMath.divFloor(baseDepth, poolQuote));
|
2020-12-13 17:47:47 +08:00
|
|
|
|
_poolI = DecimalMath.reciprocalFloor(avgPrice).mul(ratio).mul(ratio).div(
|
2020-12-12 15:53:42 +08:00
|
|
|
|
DecimalMath.ONE2
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
_POOL_ = IUnownedDVMFactory(_POOL_FACTORY_).createDODOVendingMachine(
|
2020-12-11 22:52:00 +08:00
|
|
|
|
address(this),
|
2020-12-12 15:53:42 +08:00
|
|
|
|
_poolBaseToken,
|
|
|
|
|
|
_poolQuoteToken,
|
2020-12-13 17:47:47 +08:00
|
|
|
|
3e15, // 0.3%
|
2020-12-12 15:53:42 +08:00
|
|
|
|
_poolI,
|
2020-12-11 22:52:00 +08:00
|
|
|
|
DecimalMath.ONE
|
|
|
|
|
|
);
|
2020-12-13 17:47:47 +08:00
|
|
|
|
_AVG_SETTLED_PRICE_ = avgPrice;
|
2020-12-11 22:52:00 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_transferBaseOut(_POOL_, poolBase);
|
|
|
|
|
|
_transferQuoteOut(_POOL_, poolQuote);
|
|
|
|
|
|
_transferQuoteOut(_OWNER_, ownerQuote);
|
|
|
|
|
|
|
|
|
|
|
|
IDVM(_POOL_).buyShares(address(this));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// in case something wrong with base token contract
|
|
|
|
|
|
function emergencySettle() external phaseSettlement preventReentrant {
|
2020-12-13 17:47:47 +08:00
|
|
|
|
require(block.timestamp > _PHASE_CALM_ENDTIME_.add(_SETTLEMENT_EXPIRE_), "NOT_EMERGENCY");
|
|
|
|
|
|
_settle();
|
2020-12-11 22:52:00 +08:00
|
|
|
|
_UNUSED_QUOTE_ = _QUOTE_TOKEN_.balanceOf(address(this));
|
|
|
|
|
|
_UNUSED_BASE_ = _BASE_TOKEN_.balanceOf(address(this));
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-13 17:47:47 +08:00
|
|
|
|
function _settle() internal {
|
|
|
|
|
|
require(!_SETTLED_, "ALREADY_SETTLED");
|
|
|
|
|
|
_SETTLED_ = true;
|
|
|
|
|
|
_SETTLED_TIME_ = block.timestamp;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-11 22:52:00 +08:00
|
|
|
|
// ============ Pricing ============
|
|
|
|
|
|
|
|
|
|
|
|
function getSettleResult()
|
|
|
|
|
|
public
|
|
|
|
|
|
view
|
|
|
|
|
|
returns (
|
|
|
|
|
|
uint256 poolBase,
|
|
|
|
|
|
uint256 poolQuote,
|
|
|
|
|
|
uint256 ownerQuote
|
|
|
|
|
|
)
|
|
|
|
|
|
{
|
|
|
|
|
|
poolQuote = _QUOTE_TOKEN_.balanceOf(address(this));
|
|
|
|
|
|
if (poolQuote > _POOL_QUOTE_CAP_) {
|
|
|
|
|
|
poolQuote = _POOL_QUOTE_CAP_;
|
|
|
|
|
|
}
|
|
|
|
|
|
(uint256 soldBase, ) = PMMPricing.sellQuoteToken(_getPMMState(), poolQuote);
|
|
|
|
|
|
poolBase = _TOTAL_BASE_.sub(soldBase);
|
|
|
|
|
|
ownerQuote = DecimalMath.mulFloor(poolQuote, _OWNER_QUOTE_RATIO_);
|
|
|
|
|
|
poolQuote = poolQuote.sub(ownerQuote);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function _getPMMState() internal view returns (PMMPricing.PMMState memory state) {
|
|
|
|
|
|
state.i = _I_;
|
|
|
|
|
|
state.K = _K_;
|
|
|
|
|
|
state.B = _TOTAL_BASE_;
|
|
|
|
|
|
state.Q = 0;
|
|
|
|
|
|
state.B0 = state.B;
|
|
|
|
|
|
state.Q0 = 0;
|
|
|
|
|
|
state.R = PMMPricing.RState.ONE;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-13 17:47:47 +08:00
|
|
|
|
function getExpectedAvgPrice() external view returns (uint256) {
|
|
|
|
|
|
require(!_SETTLED_, "ALREADY_SETTLED");
|
|
|
|
|
|
(uint256 poolBase, uint256 poolQuote, uint256 ownerQuote) = getSettleResult();
|
|
|
|
|
|
return
|
|
|
|
|
|
DecimalMath.divCeil(
|
|
|
|
|
|
poolQuote.add(ownerQuote),
|
|
|
|
|
|
_BASE_TOKEN_.balanceOf(address(this)).sub(poolBase)
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2020-12-11 22:52:00 +08:00
|
|
|
|
// ============ Asset In ============
|
|
|
|
|
|
|
|
|
|
|
|
function _getQuoteInput() internal view returns (uint256 input) {
|
|
|
|
|
|
return _QUOTE_TOKEN_.balanceOf(address(this)).sub(_QUOTE_RESERVE_);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ============ Set States ============
|
|
|
|
|
|
|
|
|
|
|
|
function _sync() internal {
|
|
|
|
|
|
uint256 quoteBalance = _QUOTE_TOKEN_.balanceOf(address(this));
|
|
|
|
|
|
if (quoteBalance != _QUOTE_RESERVE_) {
|
|
|
|
|
|
_QUOTE_RESERVE_ = quoteBalance;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// ============ Asset Out ============
|
|
|
|
|
|
|
|
|
|
|
|
function _transferBaseOut(address to, uint256 amount) internal {
|
|
|
|
|
|
if (amount > 0) {
|
|
|
|
|
|
_BASE_TOKEN_.safeTransfer(to, amount);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
function _transferQuoteOut(address to, uint256 amount) internal {
|
|
|
|
|
|
if (amount > 0) {
|
|
|
|
|
|
_QUOTE_TOKEN_.safeTransfer(to, amount);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2020-12-13 17:47:47 +08:00
|
|
|
|
|
|
|
|
|
|
// ============ Asset Out ============
|
|
|
|
|
|
|
|
|
|
|
|
function getShares(address user) external view returns (uint256) {
|
|
|
|
|
|
return _SHARES_[user];
|
|
|
|
|
|
}
|
2020-12-11 22:52:00 +08:00
|
|
|
|
}
|