diff --git a/contracts/Foundry.sol b/contracts/Foundry.sol index 7fe593df..7fef4dec 100644 --- a/contracts/Foundry.sol +++ b/contracts/Foundry.sol @@ -34,6 +34,26 @@ contract Foundry is IFoundry, Ownable, Initializable { meTokenRegistry = IMeTokenRegistry(_meTokenRegistry); } + // MINT FLOW CHART + /**************************************************************************** + // // + // mint() // + // | // + // CALCULATE MINT // + // / \ // + // is hub updating or meToken migrating? -{ (Y) (N) // + // / \ // + // CALCULATE | // + // TARGET MINT | // + // | | // + // TIME-WEIGHTED | // + // AVERAGE | // + // \ / // + // MINT RETURN // + // | // + // .sub(fees) // + // // + ****************************************************************************/ function mint( address _meToken, uint256 _assetsDeposited, @@ -108,6 +128,38 @@ contract Foundry is IFoundry, Ownable, Initializable { ); } + // BURN FLOW CHART + /**************************************************************************** + // // + // burn() // + // | // + // CALCULATE BURN // + // / \ // + // is hub updating or meToken migrating? -{ (Y) (N) // + // / \ // + // CALCULATE \ // + // TARGET BURN \ // + // / \ // + // TIME-WEIGHTED \ // + // AVERAGE \ // + // | | // + // WEIGHTED BURN RETURN BURN RETURN // + // / \ / \ // + // is msg.sender the -{ (N) (Y) (Y) (N) // + // owner? (vs buyer) / \ / \ // + // GET CALCULATE GET // + // TIME-WEIGHTED BALANCE LOCKED REFUND // + // REFUND RATIO RETURNED RATIO // + // | | | // + // .mul(wRR) .add(BLR) .mul(RR) // + // \_______________|_____________/ // + // | // + // ACTUAL (WEIGHTED) BURN RETURN // + // | // + // .sub(fees) // + // // + ****************************************************************************/ + /// @inheritdoc IFoundry function burn( address _meToken, @@ -125,85 +177,55 @@ contract Foundry is IFoundry, Ownable, Initializable { meToken_ = meTokenRegistry.finishResubscribe(_meToken); } // Calculate how many tokens tokens are returned - uint256 assetsReturned = calculateBurnReturn(_meToken, _meTokensBurned); + uint256 rawAssetsReturned = _calculateRawAssetsReturned( + _meToken, + _meTokensBurned + ); + uint256 assetsReturned = _calculateActualAssetsReturned( + msg.sender, + _meToken, + _meTokensBurned, + rawAssetsReturned + ); uint256 feeRate; - uint256 actualAssetsReturned; // If msg.sender == owner, give owner the sell rate. - all of tokens returned plus a % // of balancePooled based on how much % of supply will be burned // If msg.sender != owner, give msg.sender the burn rate if (msg.sender == meToken_.owner) { feeRate = fees.burnOwnerFee(); - actualAssetsReturned = - assetsReturned + - (((PRECISION * _meTokensBurned) / - IERC20(_meToken).totalSupply()) * meToken_.balanceLocked) / - PRECISION; } else { feeRate = fees.burnBuyerFee(); - if (hub_.targetRefundRatio == 0 && meToken_.targetHubId == 0) { - // Not updating targetRefundRatio or resubscribing - actualAssetsReturned = - (assetsReturned * hub_.refundRatio) / - MAX_REFUND_RATIO; - } else { - if (hub_.targetRefundRatio > 0) { - // Hub is updating - actualAssetsReturned = - (assetsReturned * - WeightedAverage.calculate( - hub_.refundRatio, - hub_.targetRefundRatio, - hub_.startTime, - hub_.endTime - )) / - MAX_REFUND_RATIO; - } else { - // meToken is resubscribing - Details.Hub memory targetHub_ = hub.getDetails( - meToken_.targetHubId - ); - actualAssetsReturned = - (assetsReturned * - WeightedAverage.calculate( - hub_.refundRatio, - targetHub_.refundRatio, - meToken_.startTime, - meToken_.endTime - )) / - MAX_REFUND_RATIO; - } - } } // Burn metoken from user IMeToken(_meToken).burn(msg.sender, _meTokensBurned); // Subtract tokens returned from balance pooled - meTokenRegistry.updateBalancePooled(false, _meToken, assetsReturned); + meTokenRegistry.updateBalancePooled(false, _meToken, rawAssetsReturned); if (msg.sender == meToken_.owner) { // Is owner, subtract from balance locked meTokenRegistry.updateBalanceLocked( false, _meToken, - actualAssetsReturned - assetsReturned + assetsReturned - rawAssetsReturned ); } else { // Is buyer, add to balance locked using refund ratio meTokenRegistry.updateBalanceLocked( true, _meToken, - assetsReturned - actualAssetsReturned + rawAssetsReturned - assetsReturned ); } - uint256 fee = actualAssetsReturned * feeRate; - actualAssetsReturned -= fee; + uint256 fee = assetsReturned * feeRate; + assetsReturned -= fee; IERC20(hub_.asset).safeTransferFrom( hub_.vault, _recipient, - actualAssetsReturned + assetsReturned ); IVault(hub_.vault).addFee(hub_.asset, fee); @@ -213,67 +235,75 @@ contract Foundry is IFoundry, Ownable, Initializable { msg.sender, _recipient, _meTokensBurned, - actualAssetsReturned + assetsReturned ); } - function viewBurn( + function calculateAssetsReturned( address _sender, address _meToken, uint256 _meTokensBurned - ) external view returns (uint256 actualAssetsReturned) { + ) external view returns (uint256 assetsReturned) { + uint256 rawAssetsReturned = _calculateRawAssetsReturned( + _meToken, + _meTokensBurned + ); + assetsReturned = _calculateActualAssetsReturned( + _sender, + _meToken, + _meTokensBurned, + rawAssetsReturned + ); + } + + function calculateAssetsDeposited( + // TODO: can we just pass in hubId instead of _meToken for first argument? + address _meToken, + uint256 _desiredMeTokensMinted + ) external view returns (uint256 assetsDeposited) { Details.MeToken memory meToken_ = meTokenRegistry.getDetails(_meToken); Details.Hub memory hub_ = hub.getDetails(meToken_.hubId); + // gas savings + uint256 totalSupply_ = IERC20(_meToken).totalSupply(); - // Calculate how many tokens tokens are returned - uint256 assetsReturned = calculateBurnReturn(_meToken, _meTokensBurned); - - uint256 feeRate; - // If msg.sender == owner, give owner the sell rate. - all of tokens returned plus a % - // of balancePooled based on how much % of supply will be burned - // If msg.sender != owner, give msg.sender the burn rate - if (_sender == meToken_.owner) { - feeRate = fees.burnOwnerFee(); - actualAssetsReturned = - assetsReturned + - (((PRECISION * _meTokensBurned) / - IERC20(_meToken).totalSupply()) * meToken_.balanceLocked) / - PRECISION; - } else { - feeRate = fees.burnBuyerFee(); - if (hub_.targetRefundRatio == 0 && meToken_.targetHubId == 0) { - // Not updating targetRefundRatio or resubscribing - actualAssetsReturned = - (assetsReturned * hub_.refundRatio) / - MAX_REFUND_RATIO; + // Calculate return assuming update is not happening + assetsDeposited = ICurve(hub_.curve).viewAssetsDeposited( + _desiredMeTokensMinted, + meToken_.hubId, + totalSupply_, + meToken_.balancePooled + ); + // Logic for if we're switching to a new curve type // updating curveDetails + if ( + (hub_.updating && (hub_.targetCurve != address(0))) || + (hub_.reconfigure) + ) { + uint256 targetAssetsDeposited; + if (hub_.targetCurve != address(0)) { + // Means we are updating to a new curve type + targetAssetsDeposited = ICurve(hub_.targetCurve) + .viewAssetsDeposited( + _desiredMeTokensMinted, + meToken_.hubId, + totalSupply_, + meToken_.balancePooled + ); } else { - if (hub_.targetRefundRatio > 0) { - // Hub is updating - actualAssetsReturned = - (assetsReturned * - WeightedAverage.calculate( - hub_.refundRatio, - hub_.targetRefundRatio, - hub_.startTime, - hub_.endTime - )) / - MAX_REFUND_RATIO; - } else { - // meToken is resubscribing - Details.Hub memory targetHub_ = hub.getDetails( - meToken_.targetHubId + // Must mean we're updating curveDetails + targetAssetsDeposited = ICurve(hub_.curve) + .viewTargetAssetsDeposited( + _desiredMeTokensMinted, + meToken_.hubId, + totalSupply_, + meToken_.balancePooled ); - actualAssetsReturned = - (assetsReturned * - WeightedAverage.calculate( - hub_.refundRatio, - targetHub_.refundRatio, - meToken_.startTime, - meToken_.endTime - )) / - MAX_REFUND_RATIO; - } } + assetsDeposited = WeightedAverage.calculate( + assetsDeposited, + targetAssetsDeposited, + hub_.startTime, + hub_.endTime + ); } } @@ -345,18 +375,17 @@ contract Foundry is IFoundry, Ownable, Initializable { } } - function calculateBurnReturn(address _meToken, uint256 _meTokensBurned) - public - view - returns (uint256 assetsReturned) - { + function _calculateRawAssetsReturned( + address _meToken, + uint256 _meTokensBurned + ) private view returns (uint256 rawAssetsReturned) { Details.MeToken memory meToken_ = meTokenRegistry.getDetails(_meToken); Details.Hub memory hub_ = hub.getDetails(meToken_.hubId); - // gas savings - uint256 totalSupply_ = IERC20(_meToken).totalSupply(); + + uint256 totalSupply_ = IERC20(_meToken).totalSupply(); // gas savings // Calculate return assuming update is not happening - assetsReturned = ICurve(hub_.curve).viewAssetsReturned( + rawAssetsReturned = ICurve(hub_.curve).viewAssetsReturned( _meTokensBurned, meToken_.hubId, totalSupply_, @@ -388,8 +417,8 @@ contract Foundry is IFoundry, Ownable, Initializable { meToken_.balancePooled ); } - assetsReturned = WeightedAverage.calculate( - assetsReturned, + rawAssetsReturned = WeightedAverage.calculate( + rawAssetsReturned, targetassetsReturned, hub_.startTime, hub_.endTime @@ -397,54 +426,58 @@ contract Foundry is IFoundry, Ownable, Initializable { } } - function calculateAssetsDeposited( - // TODO: can we just pass in hubId instead of _meToken for first argument? + /// @dev applies refundRatio + function _calculateActualAssetsReturned( + address _sender, address _meToken, - uint256 _desiredMeTokensMinted - ) public view returns (uint256 assetsDeposited) { + uint256 _meTokensBurned, + uint256 rawAssetsReturned + ) private view returns (uint256 actualAssetsReturned) { Details.MeToken memory meToken_ = meTokenRegistry.getDetails(_meToken); Details.Hub memory hub_ = hub.getDetails(meToken_.hubId); - // gas savings - uint256 totalSupply_ = IERC20(_meToken).totalSupply(); - - // Calculate return assuming update is not happening - assetsDeposited = ICurve(hub_.curve).viewAssetsDeposited( - _desiredMeTokensMinted, - meToken_.hubId, - totalSupply_, - meToken_.balancePooled - ); - // Logic for if we're switching to a new curve type // updating curveDetails - if ( - (hub_.updating && (hub_.targetCurve != address(0))) || - (hub_.reconfigure) - ) { - uint256 targetAssetsDeposited; - if (hub_.targetCurve != address(0)) { - // Means we are updating to a new curve type - targetAssetsDeposited = ICurve(hub_.targetCurve) - .viewAssetsDeposited( - _desiredMeTokensMinted, - meToken_.hubId, - totalSupply_, - meToken_.balancePooled - ); + // If msg.sender == owner, give owner the sell rate. - all of tokens returned plus a % + // of balancePooled based on how much % of supply will be burned + // If msg.sender != owner, give msg.sender the burn rate + if (_sender == meToken_.owner) { + actualAssetsReturned = + rawAssetsReturned + + (((PRECISION * _meTokensBurned) / + IERC20(_meToken).totalSupply()) * meToken_.balanceLocked) / + PRECISION; + } else { + if (hub_.targetRefundRatio == 0 && meToken_.targetHubId == 0) { + // Not updating targetRefundRatio or resubscribing + actualAssetsReturned = + (rawAssetsReturned * hub_.refundRatio) / + MAX_REFUND_RATIO; } else { - // Must mean we're updating curveDetails - targetAssetsDeposited = ICurve(hub_.curve) - .viewTargetAssetsDeposited( - _desiredMeTokensMinted, - meToken_.hubId, - totalSupply_, - meToken_.balancePooled + if (hub_.targetRefundRatio > 0) { + // Hub is updating + actualAssetsReturned = + (rawAssetsReturned * + WeightedAverage.calculate( + hub_.refundRatio, + hub_.targetRefundRatio, + hub_.startTime, + hub_.endTime + )) / + MAX_REFUND_RATIO; + } else { + // meToken is resubscribing + Details.Hub memory targetHub_ = hub.getDetails( + meToken_.targetHubId ); + actualAssetsReturned = + (rawAssetsReturned * + WeightedAverage.calculate( + hub_.refundRatio, + targetHub_.refundRatio, + meToken_.startTime, + meToken_.endTime + )) / + MAX_REFUND_RATIO; + } } - assetsDeposited = WeightedAverage.calculate( - assetsDeposited, - targetAssetsDeposited, - hub_.startTime, - hub_.endTime - ); } } } diff --git a/contracts/curves/BancorBancor.sol b/contracts/curves/BancorBancor.sol new file mode 100644 index 00000000..e65874a5 --- /dev/null +++ b/contracts/curves/BancorBancor.sol @@ -0,0 +1,348 @@ +pragma solidity ^0.8; + +import "../utils/ABDKMathQuad.sol"; +import "./Power.sol"; +import "../libs/Details.sol"; +import "../interfaces/ICurve.sol"; + +/** + * @title Bancor formula by Bancor + * @dev Modified from the original by Slava Balasanov + * https://github.com/bancorprotocol/contracts + * Split Power.sol out from BancorFormula.sol and replace SafeMath formulas with zeppelin's SafeMath + * Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements; + * and to You under the Apache License, Version 2.0. " + */ +contract BancorBancor is Power, ICurve { + using ABDKMathQuad for uint256; + using ABDKMathQuad for bytes16; + + uint32 public MAX_WEIGHT = 1000000; + bytes16 private immutable _baseX = uint256(1 ether).fromUInt(); + bytes16 private immutable _maxWeight = uint256(MAX_WEIGHT).fromUInt(); // gas savings + bytes16 private immutable _one = (uint256(1)).fromUInt(); + mapping(uint256 => Details.Bancor) private _bancors; + + function register(uint256 _hubId, bytes calldata _encodedDetails) + external + override + { + // TODO: access control + require(_encodedDetails.length > 0, "!_encodedDetails"); + + (uint256 baseY, uint32 reserveWeight) = abi.decode( + _encodedDetails, + (uint256, uint32) + ); + require(baseY > 0, "!baseY"); + require( + reserveWeight > 0 && reserveWeight <= MAX_WEIGHT, + "!reserveWeight" + ); + + Details.Bancor storage bancor_ = _bancors[_hubId]; + bancor_.baseY = baseY; + bancor_.reserveWeight = reserveWeight; + } + + function initReconfigure(uint256 _hubId, bytes calldata _encodedDetails) + external + override + { + // TODO: access control + + uint32 targetReserveWeight = abi.decode(_encodedDetails, (uint32)); + Details.Bancor storage bancor_ = _bancors[_hubId]; + + require(targetReserveWeight > 0, "!reserveWeight"); + require( + targetReserveWeight != bancor_.reserveWeight, + "targetWeight!=Weight" + ); + + // targetBaseX = (old baseY * oldR) / newR + uint256 targetBaseY = (bancor_.baseY * bancor_.reserveWeight) / + targetReserveWeight; + + bancor_.targetBaseY = targetBaseY; + bancor_.targetReserveWeight = targetReserveWeight; + } + + function finishReconfigure(uint256 _hubId) external override { + // TODO; only foundry can call + Details.Bancor storage bancor_ = _bancors[_hubId]; + bancor_.reserveWeight = bancor_.targetReserveWeight; + bancor_.baseY = bancor_.targetBaseY; + bancor_.targetReserveWeight = 0; + bancor_.targetBaseY = 0; + } + + function getDetails(uint256 bancor) + external + view + returns (Details.Bancor memory) + { + return _bancors[bancor]; + } + + function viewMeTokensMinted( + uint256 _assetsDeposited, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 meTokensMinted) { + Details.Bancor memory bancorDetails = _bancors[_hubId]; + if (_supply > 0) { + meTokensMinted = _viewMeTokensMinted( + _assetsDeposited, + bancorDetails.reserveWeight, + _supply, + _balancePooled + ); + } else { + meTokensMinted = _viewMeTokensMintedFromZero( + _assetsDeposited, + bancorDetails.reserveWeight, + bancorDetails.baseY + ); + } + } + + function viewTargetMeTokensMinted( + uint256 _assetsDeposited, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 meTokensMinted) { + Details.Bancor memory bancorDetails = _bancors[_hubId]; + if (_supply > 0) { + meTokensMinted = _viewMeTokensMinted( + _assetsDeposited, + bancorDetails.targetReserveWeight, + _supply, + _balancePooled + ); + } else { + meTokensMinted = _viewMeTokensMintedFromZero( + _assetsDeposited, + bancorDetails.targetReserveWeight, + bancorDetails.targetBaseY + ); + } + } + + function viewAssetsReturned( + uint256 _meTokensBurned, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 assetsReturned) { + Details.Bancor memory bancorDetails = _bancors[_hubId]; + assetsReturned = _viewAssetsReturned( + _meTokensBurned, + bancorDetails.reserveWeight, + _supply, + _balancePooled + ); + } + + function viewTargetAssetsReturned( + uint256 _meTokensBurned, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 assetsReturned) { + Details.Bancor memory bancorDetails = _bancors[_hubId]; + assetsReturned = _viewAssetsReturned( + _meTokensBurned, + bancorDetails.targetReserveWeight, + _supply, + _balancePooled + ); + } + + function viewAssetsDeposited( + uint256 _desiredMeTokens, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 assetsDeposited) { + Details.Bancor memory bancor_ = _bancors[_hubId]; + if (_supply > 0) { + assetsDeposited = _viewAssetsDeposited( + _desiredMeTokens, + bancor_.reserveWeight, + _supply, + bancor_.baseY, + _balancePooled + ); + } else { + assetsDeposited = _viewAssetsDepositedFromZero( + _desiredMeTokens, + bancor_.reserveWeight, + bancor_.baseY + ); + } + } + + function viewTargetAssetsDeposited( + uint256 _desiredMeTokens, + uint256 _hubId, + uint256 _supply, + uint256 _balancePooled + ) external view override returns (uint256 assetsDeposited) { + Details.Bancor memory bancor_ = _bancors[_hubId]; + if (_supply > 0) { + assetsDeposited = _viewAssetsDeposited( + _desiredMeTokens, + bancor_.targetReserveWeight, + _supply, + bancor_.targetBaseY, + _balancePooled + ); + } else { + assetsDeposited = _viewAssetsDepositedFromZero( + _desiredMeTokens, + bancor_.targetReserveWeight, + bancor_.targetBaseY + ); + } + } + + function _viewMeTokensMinted( + uint256 _depositAmount, + uint32 _connectorWeight, + uint256 _supply, + uint256 _connectorBalance + ) private view returns (uint256) { + // validate input + require( + _connectorBalance > 0 && + _connectorWeight > 0 && + _connectorWeight <= MAX_WEIGHT + ); + // special case for 0 deposit amount + if (_depositAmount == 0) { + return 0; + } + // special case if the weight = 100% + if (_connectorWeight == MAX_WEIGHT) { + // TODO: will ABDK round correctly here? + return (_supply * _depositAmount) / _connectorBalance; + } + uint256 result; + uint8 precision; + uint256 baseN = _depositAmount + _connectorBalance; + (result, precision) = power( + baseN, + _connectorBalance, + _connectorWeight, + MAX_WEIGHT + ); + // TODO: will ABDK shift correctly here? + uint256 newTokenSupply = (_supply * result) >> precision; + return newTokenSupply - _supply; + } + + function _viewMeTokensMintedFromZero( + uint256 _assetsDeposited, + uint256 _reserveWeight, + uint256 _baseY + ) private view returns (uint256) { + bytes16 reserveWeight = _reserveWeight.fromUInt().div(_maxWeight); + // _assetsDeposited * baseY ^ (1/connectorWeight) + bytes16 numerator = _assetsDeposited.fromUInt().mul( + _baseY.fromUInt().ln().mul(_one.div(reserveWeight)).exp() + ); + // as baseX == 1 ether and we want to result to be in ether too we simply remove + // the multiplication by baseY + bytes16 denominator = reserveWeight.mul(_baseY.fromUInt()); + // Instead of calculating "x ^ exp", we calculate "e ^ (log(x) * exp)". + // (numerator/denominator) ^ (reserveWeight ) + // => e^ (log(numerator/denominator) * reserveWeight ) + // => log(numerator/denominator) == (numerator.div(denominator)).ln() + // => (numerator.div(denominator)).ln().mul(reserveWeight).exp(); + bytes16 res = (numerator.div(denominator)) + .ln() + .mul(reserveWeight) + .exp() + .mul(_maxWeight); + return res.toUInt(); + } + + function _viewAssetsReturned( + uint256 _sellAmount, + uint32 _connectorWeight, + uint256 _supply, + uint256 _connectorBalance + ) private view returns (uint256) { + // validate input + require( + _supply > 0 && + _connectorBalance > 0 && + _connectorWeight > 0 && + _connectorWeight <= MAX_WEIGHT && + _sellAmount <= _supply + ); + // special case for 0 sell amount + if (_sellAmount == 0) { + return 0; + } + // special case for selling the entire supply + if (_sellAmount == _supply) { + return _connectorBalance; + } + // special case if the weight = 100% + if (_connectorWeight == MAX_WEIGHT) { + return (_connectorBalance * _sellAmount) / _supply; + } + uint256 result; + uint8 precision; + uint256 baseD = _supply - _sellAmount; + (result, precision) = power( + _supply, + baseD, + MAX_WEIGHT, + _connectorWeight + ); + uint256 oldBalance = _connectorBalance * result; + uint256 newBalance = _connectorBalance << precision; + return (oldBalance - newBalance) / result; + } + + function _viewAssetsDepositedFromZero( + uint256 _desiredMeTokens, + uint256 _reserveWeight, + uint256 _baseY + ) private view returns (uint256) { + bytes16 reserveWeight = _reserveWeight.fromUInt().div(_maxWeight); + bytes16 numerator = _baseY.fromUInt().mul(reserveWeight); + // Instead of calculating s ^ exp, we calculate e ^ (log(s) * exp). + bytes16 squared = _desiredMeTokens + .fromUInt() + .ln() + .mul(uint256(2).fromUInt()) + .exp(); + bytes16 res = numerator.mul(squared).div(_baseX); + return res.toUInt(); + } + + function _viewAssetsDeposited( + uint256 _desiredMeTokens, + uint256 _reserveWeight, + uint256 _supply, + uint256 _baseY, + uint256 _balancePooled + ) private view returns (uint256) { + // TODO: does this need to be divided? + bytes16 reserveWeight = _reserveWeight.fromUInt().div(_maxWeight); + bytes16 k = _baseY.fromUInt().mul(reserveWeight).div(_baseX); + bytes16 squared = (_supply + _desiredMeTokens) + .fromUInt() + .ln() + .mul(uint256(2).fromUInt()) + .exp(); + bytes16 res = k.mul(squared).sub(_balancePooled.fromUInt()); + return res.toUInt(); + } +} diff --git a/contracts/curves/BancorZeroCurve.sol b/contracts/curves/BancorZeroCurve.sol index dedd35ae..5c59fe09 100644 --- a/contracts/curves/BancorZeroCurve.sol +++ b/contracts/curves/BancorZeroCurve.sol @@ -212,6 +212,20 @@ contract BancorZeroCurve is ICurve { } } + ///************************* CALCULATE FUNCTIONS **************************/ + ///**************** - USED BY MINT & BURN IN FOUNDRY.SOL - ****************/ + + // CALCULATE MINT + /*************************************************************************** + // __ __ // + // T = meTokensReturned / \ (rW) // + // D = depositAmount | 1 + D | ^ // + // rW = reserveWeight T = S * | ---------- | - 1 // + // bP = balancePooled | bP | // + // S = supply \__ __/ // + // // + ***************************************************************************/ + /// @notice Given a deposit (in the connector token), reserve weight, meToken supply and /// balance pooled, calculate the return for a given conversion (in the meToken) /// @dev _supply * ((1 + _assetsDeposited / _balancePooled) ^ (_reserveWeight / 1000000) - 1) @@ -254,6 +268,17 @@ contract BancorZeroCurve is ICurve { return res.toUInt(); } + // CALCULATE MINT (FROM ZERO) + /*************************************************************************** + // __ __ // + // T = meTokensReturned / (1/rW) \ (rW) // + // D = depositAmouont | D * y ^ | ^ // + // rW = reserveWeight T = | ---------------- | // + // x = baseX | rW * x * y | // + // y = baseY \__ __/ // + // // + ***************************************************************************/ + /// @notice Given a deposit (in the collateral token) meToken supply of 0, constant x and /// constant y, calculates the return for a given conversion (in the meToken) /// @dev _baseX / (_baseY ^ (MAX_WEIGHT/reserveWeight -1)) * assetsDeposited ^(MAX_WEIGHT/reserveWeight -1) @@ -286,6 +311,17 @@ contract BancorZeroCurve is ICurve { return res.toUInt(); } + // CALCULATE BURN + /**************************************************************************** + // __ __ // + // T = tokensReturned / \ (1/rW) // + // B = meTokensBurned | 1 + B | ^ // + // rW = reserveWeight T = rB * | ---------- | - 1 // + // bP = balancePooled | s | // + // S = supply \__ __/ // + // // + ****************************************************************************/ + /// @notice Given an amount of meTokens to burn, connector weight, supply and collateral pooled, /// calculates the return for a given conversion (in the collateral token) /// @dev _balancePooled * (1 - (1 - _meTokensBurned/_supply) ^ (1 / (_reserveWeight / 1000000))) @@ -336,24 +372,21 @@ contract BancorZeroCurve is ICurve { return res.toUInt(); } - // (baseY * desiredMeTokens^2 * reserveWeight) / baseX - // Or (baseY * reserveWeight) / baseX * desiredMeTokens^2 - function _viewAssetsDepositedFromZero( - uint256 _desiredMeTokens, - uint256 _reserveWeight, - uint256 _baseY - ) private view returns (uint256) { - bytes16 reserveWeight = _reserveWeight.fromUInt().div(_maxWeight); - bytes16 numerator = _baseY.fromUInt().mul(reserveWeight); - // Instead of calculating s ^ exp, we calculate e ^ (log(s) * exp). - bytes16 squared = _desiredMeTokens - .fromUInt() - .ln() - .mul(uint256(2).fromUInt()) - .exp(); - bytes16 res = numerator.mul(squared).div(_baseX); - return res.toUInt(); - } + ///*************************** VIEW FUNCTIONS *****************************/ + ///******* - USED FOR ORACLES & THE FRONT END VIA FOUNDRY.SOL - ***********/ + + // VIEW COLLATERAL FOR DESIRED MINT + /*************************************************************************** + // __ __ // + // T = tokensToDeposit / \ // + // d = desiredMeTokensMinted | y * rW | // + // rW = reserveWeight T = | -------------- | - bP // + // bP = balancePooled | x * (S + d)^2 | // + // S = supply \__ __/ // + // x = baseX // + // y = baseY // + // // + ***************************************************************************/ // baseY * (supply + desiredMeTokens)^2 * reserveWeight / baseX - balancePooled // or (baseY * reserveWeight) / baseX * (supply + desiredMeTokens)^2 - balancePooled @@ -375,4 +408,47 @@ contract BancorZeroCurve is ICurve { bytes16 res = k.mul(squared).sub(_balancePooled.fromUInt()); return res.toUInt(); } + + // VIEW COLLATERAL FOR DESIRED MINT (FROM ZERO) + /*************************************************************************** + // // + // T = tokensToDeposit // + // d = desiredMeTokensMinted y * rW // + // rW = reserveWeight T = --------- // + // x = baseX x * d^2 // + // y = baseY // + // // + ***************************************************************************/ + + // (baseY * desiredMeTokens^2 * reserveWeight) / baseX + // Or (baseY * reserveWeight) / baseX * desiredMeTokens^2 + function _viewAssetsDepositedFromZero( + uint256 _desiredMeTokens, + uint256 _reserveWeight, + uint256 _baseY + ) private view returns (uint256) { + bytes16 reserveWeight = _reserveWeight.fromUInt().div(_maxWeight); + bytes16 numerator = _baseY.fromUInt().mul(reserveWeight); + // Instead of calculating s ^ exp, we calculate e ^ (log(s) * exp). + bytes16 squared = _desiredMeTokens + .fromUInt() + .ln() + .mul(uint256(2).fromUInt()) + .exp(); + bytes16 res = numerator.mul(squared).div(_baseX); + return res.toUInt(); + } + + // VIEW BURN FOR DESIRED COLLATERAL RETURNED + /*************************************************************************** + // __ __ // + // T = meTokensToBurn / \ // + // d = desiredCollateraReturned | | // + // rW = reserveWeight T = | ---------- | // + // bP = balancePooled | | // + // S = supply \__ __/ // + // // + ***************************************************************************/ + + // TODO: Create function } diff --git a/contracts/curves/Power.sol b/contracts/curves/Power.sol new file mode 100644 index 00000000..db40b57d --- /dev/null +++ b/contracts/curves/Power.sol @@ -0,0 +1,544 @@ +pragma solidity ^0.8; + +/** + * @title Power function by Bancor + * @dev https://github.com/bancorprotocol/contracts + * + * Modified from the original by Slava Balasanov & Tarrence van As + * + * Split Power.sol out from BancorFormula.sol + */ +contract Power { + string public version = "0.3"; + + uint256 private constant ONE = 1; + uint8 private constant MIN_PRECISION = 32; + uint8 private constant MAX_PRECISION = 127; + + uint256 private constant FIXED_1 = 0x080000000000000000000000000000000; + uint256 private constant FIXED_2 = 0x100000000000000000000000000000000; + uint256 private constant MAX_NUM = 0x200000000000000000000000000000000; + + uint256 private constant LN2_NUMERATOR = 0x3f80fe03f80fe03f80fe03f80fe03f8; + uint256 private constant LN2_DENOMINATOR = + 0x5b9de1d10bf4103d647b0955897ba80; + + uint256 private constant OPT_LOG_MAX_VAL = + 0x15bf0a8b1457695355fb8ac404e7a79e3; + uint256 private constant OPT_EXP_MAX_VAL = + 0x800000000000000000000000000000000; + + uint256[128] private maxExpArray; + + constructor() { + // maxExpArray[0] = 0x6bffffffffffffffffffffffffffffffff; + // maxExpArray[1] = 0x67ffffffffffffffffffffffffffffffff; + // maxExpArray[2] = 0x637fffffffffffffffffffffffffffffff; + // maxExpArray[3] = 0x5f6fffffffffffffffffffffffffffffff; + // maxExpArray[4] = 0x5b77ffffffffffffffffffffffffffffff; + // maxExpArray[5] = 0x57b3ffffffffffffffffffffffffffffff; + // maxExpArray[6] = 0x5419ffffffffffffffffffffffffffffff; + // maxExpArray[7] = 0x50a2ffffffffffffffffffffffffffffff; + // maxExpArray[8] = 0x4d517fffffffffffffffffffffffffffff; + // maxExpArray[9] = 0x4a233fffffffffffffffffffffffffffff; + // maxExpArray[10] = 0x47165fffffffffffffffffffffffffffff; + // maxExpArray[11] = 0x4429afffffffffffffffffffffffffffff; + // maxExpArray[12] = 0x415bc7ffffffffffffffffffffffffffff; + // maxExpArray[13] = 0x3eab73ffffffffffffffffffffffffffff; + // maxExpArray[14] = 0x3c1771ffffffffffffffffffffffffffff; + // maxExpArray[15] = 0x399e96ffffffffffffffffffffffffffff; + // maxExpArray[16] = 0x373fc47fffffffffffffffffffffffffff; + // maxExpArray[17] = 0x34f9e8ffffffffffffffffffffffffffff; + // maxExpArray[18] = 0x32cbfd5fffffffffffffffffffffffffff; + // maxExpArray[19] = 0x30b5057fffffffffffffffffffffffffff; + // maxExpArray[20] = 0x2eb40f9fffffffffffffffffffffffffff; + // maxExpArray[21] = 0x2cc8340fffffffffffffffffffffffffff; + // maxExpArray[22] = 0x2af09481ffffffffffffffffffffffffff; + // maxExpArray[23] = 0x292c5bddffffffffffffffffffffffffff; + // maxExpArray[24] = 0x277abdcdffffffffffffffffffffffffff; + // maxExpArray[25] = 0x25daf6657fffffffffffffffffffffffff; + // maxExpArray[26] = 0x244c49c65fffffffffffffffffffffffff; + // maxExpArray[27] = 0x22ce03cd5fffffffffffffffffffffffff; + // maxExpArray[28] = 0x215f77c047ffffffffffffffffffffffff; + // maxExpArray[29] = 0x1fffffffffffffffffffffffffffffffff; + // maxExpArray[30] = 0x1eaefdbdabffffffffffffffffffffffff; + // maxExpArray[31] = 0x1d6bd8b2ebffffffffffffffffffffffff; + maxExpArray[32] = 0x1c35fedd14ffffffffffffffffffffffff; + maxExpArray[33] = 0x1b0ce43b323fffffffffffffffffffffff; + maxExpArray[34] = 0x19f0028ec1ffffffffffffffffffffffff; + maxExpArray[35] = 0x18ded91f0e7fffffffffffffffffffffff; + maxExpArray[36] = 0x17d8ec7f0417ffffffffffffffffffffff; + maxExpArray[37] = 0x16ddc6556cdbffffffffffffffffffffff; + maxExpArray[38] = 0x15ecf52776a1ffffffffffffffffffffff; + maxExpArray[39] = 0x15060c256cb2ffffffffffffffffffffff; + maxExpArray[40] = 0x1428a2f98d72ffffffffffffffffffffff; + maxExpArray[41] = 0x13545598e5c23fffffffffffffffffffff; + maxExpArray[42] = 0x1288c4161ce1dfffffffffffffffffffff; + maxExpArray[43] = 0x11c592761c666fffffffffffffffffffff; + maxExpArray[44] = 0x110a688680a757ffffffffffffffffffff; + maxExpArray[45] = 0x1056f1b5bedf77ffffffffffffffffffff; + maxExpArray[46] = 0x0faadceceeff8bffffffffffffffffffff; + maxExpArray[47] = 0x0f05dc6b27edadffffffffffffffffffff; + maxExpArray[48] = 0x0e67a5a25da4107fffffffffffffffffff; + maxExpArray[49] = 0x0dcff115b14eedffffffffffffffffffff; + maxExpArray[50] = 0x0d3e7a392431239fffffffffffffffffff; + maxExpArray[51] = 0x0cb2ff529eb71e4fffffffffffffffffff; + maxExpArray[52] = 0x0c2d415c3db974afffffffffffffffffff; + maxExpArray[53] = 0x0bad03e7d883f69bffffffffffffffffff; + maxExpArray[54] = 0x0b320d03b2c343d5ffffffffffffffffff; + maxExpArray[55] = 0x0abc25204e02828dffffffffffffffffff; + maxExpArray[56] = 0x0a4b16f74ee4bb207fffffffffffffffff; + maxExpArray[57] = 0x09deaf736ac1f569ffffffffffffffffff; + maxExpArray[58] = 0x0976bd9952c7aa957fffffffffffffffff; + maxExpArray[59] = 0x09131271922eaa606fffffffffffffffff; + maxExpArray[60] = 0x08b380f3558668c46fffffffffffffffff; + maxExpArray[61] = 0x0857ddf0117efa215bffffffffffffffff; + maxExpArray[62] = 0x07ffffffffffffffffffffffffffffffff; + maxExpArray[63] = 0x07abbf6f6abb9d087fffffffffffffffff; + maxExpArray[64] = 0x075af62cbac95f7dfa7fffffffffffffff; + maxExpArray[65] = 0x070d7fb7452e187ac13fffffffffffffff; + maxExpArray[66] = 0x06c3390ecc8af379295fffffffffffffff; + maxExpArray[67] = 0x067c00a3b07ffc01fd6fffffffffffffff; + maxExpArray[68] = 0x0637b647c39cbb9d3d27ffffffffffffff; + maxExpArray[69] = 0x05f63b1fc104dbd39587ffffffffffffff; + maxExpArray[70] = 0x05b771955b36e12f7235ffffffffffffff; + maxExpArray[71] = 0x057b3d49dda84556d6f6ffffffffffffff; + maxExpArray[72] = 0x054183095b2c8ececf30ffffffffffffff; + maxExpArray[73] = 0x050a28be635ca2b888f77fffffffffffff; + maxExpArray[74] = 0x04d5156639708c9db33c3fffffffffffff; + maxExpArray[75] = 0x04a23105873875bd52dfdfffffffffffff; + maxExpArray[76] = 0x0471649d87199aa990756fffffffffffff; + maxExpArray[77] = 0x04429a21a029d4c1457cfbffffffffffff; + maxExpArray[78] = 0x0415bc6d6fb7dd71af2cb3ffffffffffff; + maxExpArray[79] = 0x03eab73b3bbfe282243ce1ffffffffffff; + maxExpArray[80] = 0x03c1771ac9fb6b4c18e229ffffffffffff; + maxExpArray[81] = 0x0399e96897690418f785257fffffffffff; + maxExpArray[82] = 0x0373fc456c53bb779bf0ea9fffffffffff; + maxExpArray[83] = 0x034f9e8e490c48e67e6ab8bfffffffffff; + maxExpArray[84] = 0x032cbfd4a7adc790560b3337ffffffffff; + maxExpArray[85] = 0x030b50570f6e5d2acca94613ffffffffff; + maxExpArray[86] = 0x02eb40f9f620fda6b56c2861ffffffffff; + maxExpArray[87] = 0x02cc8340ecb0d0f520a6af58ffffffffff; + maxExpArray[88] = 0x02af09481380a0a35cf1ba02ffffffffff; + maxExpArray[89] = 0x0292c5bdd3b92ec810287b1b3fffffffff; + maxExpArray[90] = 0x0277abdcdab07d5a77ac6d6b9fffffffff; + maxExpArray[91] = 0x025daf6654b1eaa55fd64df5efffffffff; + maxExpArray[92] = 0x0244c49c648baa98192dce88b7ffffffff; + maxExpArray[93] = 0x022ce03cd5619a311b2471268bffffffff; + maxExpArray[94] = 0x0215f77c045fbe885654a44a0fffffffff; + maxExpArray[95] = 0x01ffffffffffffffffffffffffffffffff; + maxExpArray[96] = 0x01eaefdbdaaee7421fc4d3ede5ffffffff; + maxExpArray[97] = 0x01d6bd8b2eb257df7e8ca57b09bfffffff; + maxExpArray[98] = 0x01c35fedd14b861eb0443f7f133fffffff; + maxExpArray[99] = 0x01b0ce43b322bcde4a56e8ada5afffffff; + maxExpArray[100] = 0x019f0028ec1fff007f5a195a39dfffffff; + maxExpArray[101] = 0x018ded91f0e72ee74f49b15ba527ffffff; + maxExpArray[102] = 0x017d8ec7f04136f4e5615fd41a63ffffff; + maxExpArray[103] = 0x016ddc6556cdb84bdc8d12d22e6fffffff; + maxExpArray[104] = 0x015ecf52776a1155b5bd8395814f7fffff; + maxExpArray[105] = 0x015060c256cb23b3b3cc3754cf40ffffff; + maxExpArray[106] = 0x01428a2f98d728ae223ddab715be3fffff; + maxExpArray[107] = 0x013545598e5c23276ccf0ede68034fffff; + maxExpArray[108] = 0x01288c4161ce1d6f54b7f61081194fffff; + maxExpArray[109] = 0x011c592761c666aa641d5a01a40f17ffff; + maxExpArray[110] = 0x0110a688680a7530515f3e6e6cfdcdffff; + maxExpArray[111] = 0x01056f1b5bedf75c6bcb2ce8aed428ffff; + maxExpArray[112] = 0x00faadceceeff8a0890f3875f008277fff; + maxExpArray[113] = 0x00f05dc6b27edad306388a600f6ba0bfff; + maxExpArray[114] = 0x00e67a5a25da41063de1495d5b18cdbfff; + maxExpArray[115] = 0x00dcff115b14eedde6fc3aa5353f2e4fff; + maxExpArray[116] = 0x00d3e7a3924312399f9aae2e0f868f8fff; + maxExpArray[117] = 0x00cb2ff529eb71e41582cccd5a1ee26fff; + maxExpArray[118] = 0x00c2d415c3db974ab32a51840c0b67edff; + maxExpArray[119] = 0x00bad03e7d883f69ad5b0a186184e06bff; + maxExpArray[120] = 0x00b320d03b2c343d4829abd6075f0cc5ff; + maxExpArray[121] = 0x00abc25204e02828d73c6e80bcdb1a95bf; + maxExpArray[122] = 0x00a4b16f74ee4bb2040a1ec6c15fbbf2df; + maxExpArray[123] = 0x009deaf736ac1f569deb1b5ae3f36c130f; + maxExpArray[124] = 0x00976bd9952c7aa957f5937d790ef65037; + maxExpArray[125] = 0x009131271922eaa6064b73a22d0bd4f2bf; + maxExpArray[126] = 0x008b380f3558668c46c91c49a2f8e967b9; + maxExpArray[127] = 0x00857ddf0117efa215952912839f6473e6; + } + + /** + General Description: + Determine a value of precision. + Calculate an integer approximation of (_baseN / _baseD) ^ (_expN / _expD) * 2 ^ precision. + Return the result along with the precision used. + Detailed Description: + Instead of calculating "base ^ exp", we calculate "e ^ (log(base) * exp)". + The value of "log(base)" is represented with an integer slightly smaller than "log(base) * 2 ^ precision". + The larger "precision" is, the more accurately this value represents the real value. + However, the larger "precision" is, the more bits are required in order to store this value. + And the exponentiation function, which takes "x" and calculates "e ^ x", is limited to a maximum exponent (maximum value of "x"). + This maximum exponent depends on the "precision" used, and it is given by "maxExpArray[precision] >> (MAX_PRECISION - precision)". + Hence we need to determine the highest precision which can be used for the given input, before calling the exponentiation function. + This allows us to compute "base ^ exp" with maximum accuracy and without exceeding 256 bits in any of the intermediate computations. + This functions assumes that "_expN < 2 ^ 256 / log(MAX_NUM - 1)", otherwise the multiplication should be replaced with a "safeMul". + */ + function power( + uint256 _baseN, + uint256 _baseD, + uint32 _expN, + uint32 _expD + ) internal view returns (uint256, uint8) { + assert(_baseN < MAX_NUM); + require(_baseN >= _baseD, "Bases < 1 are not supported."); + + uint256 baseLog; + uint256 base = (_baseN * FIXED_1) / _baseD; + if (base < OPT_LOG_MAX_VAL) { + baseLog = optimalLog(base); + } else { + baseLog = generalLog(base); + } + + uint256 baseLogTimesExp = (baseLog * _expN) / _expD; + if (baseLogTimesExp < OPT_EXP_MAX_VAL) { + return (optimalExp(baseLogTimesExp), MAX_PRECISION); + } else { + uint8 precision = findPositionInMaxExpArray(baseLogTimesExp); + return ( + generalExp( + baseLogTimesExp >> (MAX_PRECISION - precision), + precision + ), + precision + ); + } + } + + /** + Compute log(x / FIXED_1) * FIXED_1. + This functions assumes that "x >= FIXED_1", because the output would be negative otherwise. + */ + function generalLog(uint256 _x) internal pure returns (uint256) { + uint256 res = 0; + uint256 x = _x; + + // If x >= 2, then we compute the integer part of log2(x), which is larger than 0. + if (x >= FIXED_2) { + uint8 count = floorLog2(x / FIXED_1); + x >>= count; // now x < 2 + res = count * FIXED_1; + } + + // If x > 1, then we compute the fraction part of log2(x), which is larger than 0. + if (x > FIXED_1) { + for (uint8 i = MAX_PRECISION; i > 0; --i) { + x = (x * x) / FIXED_1; // now 1 < x < 4 + if (x >= FIXED_2) { + x >>= 1; // now 1 < x < 2 + res += ONE << (i - 1); + } + } + } + + return (res * LN2_NUMERATOR) / LN2_DENOMINATOR; + } + + /** + Compute the largest integer smaller than or equal to the binary logarithm of the input. + */ + function floorLog2(uint256 _n) internal pure returns (uint8) { + uint8 res = 0; + uint256 n = _n; + + if (n < 256) { + // At most 8 iterations + while (n > 1) { + n >>= 1; + res += 1; + } + } else { + // Exactly 8 iterations + for (uint8 s = 128; s > 0; s >>= 1) { + if (n >= (ONE << s)) { + n >>= s; + res |= s; + } + } + } + + return res; + } + + /** + The global "maxExpArray" is sorted in descending order, and therefore the following statements are equivalent: + - This function finds the position of [the smallest value in "maxExpArray" larger than or equal to "x"] + - This function finds the highest position of [a value in "maxExpArray" larger than or equal to "x"] + */ + function findPositionInMaxExpArray(uint256 _x) + internal + view + returns (uint8) + { + uint8 lo = MIN_PRECISION; + uint8 hi = MAX_PRECISION; + + while (lo + 1 < hi) { + uint8 mid = (lo + hi) / 2; + if (maxExpArray[mid] >= _x) lo = mid; + else hi = mid; + } + + if (maxExpArray[hi] >= _x) return hi; + if (maxExpArray[lo] >= _x) return lo; + + assert(false); + return 0; + } + + /* solium-disable */ + /** + This function can be auto-generated by the script 'PrintFunctionGeneralExp.py'. + It approximates "e ^ x" via maclaurin summation: "(x^0)/0! + (x^1)/1! + ... + (x^n)/n!". + It returns "e ^ (x / 2 ^ precision) * 2 ^ precision", that is, the result is upshifted for accuracy. + The global "maxExpArray" maps each "precision" to "((maximumExponent + 1) << (MAX_PRECISION - precision)) - 1". + The maximum permitted value for "x" is therefore given by "maxExpArray[precision] >> (MAX_PRECISION - precision)". + */ + function generalExp(uint256 _x, uint8 _precision) + internal + pure + returns (uint256) + { + uint256 xi = _x; + uint256 res = 0; + + xi = (xi * _x) >> _precision; + res += xi * 0x3442c4e6074a82f1797f72ac0000000; // add x^02 * (33! / 02!) + xi = (xi * _x) >> _precision; + res += xi * 0x116b96f757c380fb287fd0e40000000; // add x^03 * (33! / 03!) + xi = (xi * _x) >> _precision; + res += xi * 0x045ae5bdd5f0e03eca1ff4390000000; // add x^04 * (33! / 04!) + xi = (xi * _x) >> _precision; + res += xi * 0x00defabf91302cd95b9ffda50000000; // add x^05 * (33! / 05!) + xi = (xi * _x) >> _precision; + res += xi * 0x002529ca9832b22439efff9b8000000; // add x^06 * (33! / 06!) + xi = (xi * _x) >> _precision; + res += xi * 0x00054f1cf12bd04e516b6da88000000; // add x^07 * (33! / 07!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000a9e39e257a09ca2d6db51000000; // add x^08 * (33! / 08!) + xi = (xi * _x) >> _precision; + res += xi * 0x000012e066e7b839fa050c309000000; // add x^09 * (33! / 09!) + xi = (xi * _x) >> _precision; + res += xi * 0x000001e33d7d926c329a1ad1a800000; // add x^10 * (33! / 10!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000002bee513bdb4a6b19b5f800000; // add x^11 * (33! / 11!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000003a9316fa79b88eccf2a00000; // add x^12 * (33! / 12!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000048177ebe1fa812375200000; // add x^13 * (33! / 13!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000005263fe90242dcbacf00000; // add x^14 * (33! / 14!) + xi = (xi * _x) >> _precision; + res += xi * 0x000000000057e22099c030d94100000; // add x^15 * (33! / 15!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000057e22099c030d9410000; // add x^16 * (33! / 16!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000052b6b54569976310000; // add x^17 * (33! / 17!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000004985f67696bf748000; // add x^18 * (33! / 18!) + xi = (xi * _x) >> _precision; + res += xi * 0x000000000000003dea12ea99e498000; // add x^19 * (33! / 19!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000000031880f2214b6e000; // add x^20 * (33! / 20!) + xi = (xi * _x) >> _precision; + res += xi * 0x000000000000000025bcff56eb36000; // add x^21 * (33! / 21!) + xi = (xi * _x) >> _precision; + res += xi * 0x000000000000000001b722e10ab1000; // add x^22 * (33! / 22!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000001317c70077000; // add x^23 * (33! / 23!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000000000000cba84aafa00; // add x^24 * (33! / 24!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000000000000082573a0a00; // add x^25 * (33! / 25!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000000000000005035ad900; // add x^26 * (33! / 26!) + xi = (xi * _x) >> _precision; + res += xi * 0x000000000000000000000002f881b00; // add x^27 * (33! / 27!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000000000001b29340; // add x^28 * (33! / 28!) + xi = (xi * _x) >> _precision; + res += xi * 0x00000000000000000000000000efc40; // add x^29 * (33! / 29!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000000000000007fe0; // add x^30 * (33! / 30!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000000000000000420; // add x^31 * (33! / 31!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000000000000000021; // add x^32 * (33! / 32!) + xi = (xi * _x) >> _precision; + res += xi * 0x0000000000000000000000000000001; // add x^33 * (33! / 33!) + + return + res / 0x688589cc0e9505e2f2fee5580000000 + _x + (ONE << _precision); // divide by 33! and then add x^1 / 1! + x^0 / 0! + } + + /** + Return log(x / FIXED_1) * FIXED_1 + Input range: FIXED_1 <= x <= LOG_EXP_MAX_VAL - 1 + Auto-generated via 'PrintFunctionOptimalLog.py' + */ + function optimalLog(uint256 x) internal pure returns (uint256) { + uint256 res = 0; + + uint256 y; + uint256 z; + uint256 w; + + if (x >= 0xd3094c70f034de4b96ff7d5b6f99fcd8) { + res += 0x40000000000000000000000000000000; + x = (x * FIXED_1) / 0xd3094c70f034de4b96ff7d5b6f99fcd8; + } + if (x >= 0xa45af1e1f40c333b3de1db4dd55f29a7) { + res += 0x20000000000000000000000000000000; + x = (x * FIXED_1) / 0xa45af1e1f40c333b3de1db4dd55f29a7; + } + if (x >= 0x910b022db7ae67ce76b441c27035c6a1) { + res += 0x10000000000000000000000000000000; + x = (x * FIXED_1) / 0x910b022db7ae67ce76b441c27035c6a1; + } + if (x >= 0x88415abbe9a76bead8d00cf112e4d4a8) { + res += 0x08000000000000000000000000000000; + x = (x * FIXED_1) / 0x88415abbe9a76bead8d00cf112e4d4a8; + } + if (x >= 0x84102b00893f64c705e841d5d4064bd3) { + res += 0x04000000000000000000000000000000; + x = (x * FIXED_1) / 0x84102b00893f64c705e841d5d4064bd3; + } + if (x >= 0x8204055aaef1c8bd5c3259f4822735a2) { + res += 0x02000000000000000000000000000000; + x = (x * FIXED_1) / 0x8204055aaef1c8bd5c3259f4822735a2; + } + if (x >= 0x810100ab00222d861931c15e39b44e99) { + res += 0x01000000000000000000000000000000; + x = (x * FIXED_1) / 0x810100ab00222d861931c15e39b44e99; + } + if (x >= 0x808040155aabbbe9451521693554f733) { + res += 0x00800000000000000000000000000000; + x = (x * FIXED_1) / 0x808040155aabbbe9451521693554f733; + } + + z = y = x - FIXED_1; + w = (y * y) / FIXED_1; + res += + (z * (0x100000000000000000000000000000000 - y)) / + 0x100000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x0aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa - y)) / + 0x200000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x099999999999999999999999999999999 - y)) / + 0x300000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x092492492492492492492492492492492 - y)) / + 0x400000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x08e38e38e38e38e38e38e38e38e38e38e - y)) / + 0x500000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x08ba2e8ba2e8ba2e8ba2e8ba2e8ba2e8b - y)) / + 0x600000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x089d89d89d89d89d89d89d89d89d89d89 - y)) / + 0x700000000000000000000000000000000; + z = (z * w) / FIXED_1; + res += + (z * (0x088888888888888888888888888888888 - y)) / + 0x800000000000000000000000000000000; + + return res; + } + + /** + Return e ^ (x / FIXED_1) * FIXED_1 + Input range: 0 <= x <= OPT_EXP_MAX_VAL - 1 + Auto-generated via 'PrintFunctionOptimalExp.py' + */ + function optimalExp(uint256 x) internal pure returns (uint256) { + uint256 res = 0; + + uint256 y; + uint256 z; + + z = y = x % 0x10000000000000000000000000000000; + z = (z * y) / FIXED_1; + res += z * 0x10e1b3be415a0000; // add y^02 * (20! / 02!) + z = (z * y) / FIXED_1; + res += z * 0x05a0913f6b1e0000; // add y^03 * (20! / 03!) + z = (z * y) / FIXED_1; + res += z * 0x0168244fdac78000; // add y^04 * (20! / 04!) + z = (z * y) / FIXED_1; + res += z * 0x004807432bc18000; // add y^05 * (20! / 05!) + z = (z * y) / FIXED_1; + res += z * 0x000c0135dca04000; // add y^06 * (20! / 06!) + z = (z * y) / FIXED_1; + res += z * 0x0001b707b1cdc000; // add y^07 * (20! / 07!) + z = (z * y) / FIXED_1; + res += z * 0x000036e0f639b800; // add y^08 * (20! / 08!) + z = (z * y) / FIXED_1; + res += z * 0x00000618fee9f800; // add y^09 * (20! / 09!) + z = (z * y) / FIXED_1; + res += z * 0x0000009c197dcc00; // add y^10 * (20! / 10!) + z = (z * y) / FIXED_1; + res += z * 0x0000000e30dce400; // add y^11 * (20! / 11!) + z = (z * y) / FIXED_1; + res += z * 0x000000012ebd1300; // add y^12 * (20! / 12!) + z = (z * y) / FIXED_1; + res += z * 0x0000000017499f00; // add y^13 * (20! / 13!) + z = (z * y) / FIXED_1; + res += z * 0x0000000001a9d480; // add y^14 * (20! / 14!) + z = (z * y) / FIXED_1; + res += z * 0x00000000001c6380; // add y^15 * (20! / 15!) + z = (z * y) / FIXED_1; + res += z * 0x000000000001c638; // add y^16 * (20! / 16!) + z = (z * y) / FIXED_1; + res += z * 0x0000000000001ab8; // add y^17 * (20! / 17!) + z = (z * y) / FIXED_1; + res += z * 0x000000000000017c; // add y^18 * (20! / 18!) + z = (z * y) / FIXED_1; + res += z * 0x0000000000000014; // add y^19 * (20! / 19!) + z = (z * y) / FIXED_1; + res += z * 0x0000000000000001; // add y^20 * (20! / 20!) + res = res / 0x21c3677c82b40000 + y + FIXED_1; // divide by 20! and then add y^1 / 1! + y^0 / 0! + + if ((x & 0x010000000000000000000000000000000) != 0) + res = + (res * 0x1c3d6a24ed82218787d624d3e5eba95f9) / + 0x18ebef9eac820ae8682b9793ac6d1e776; + if ((x & 0x020000000000000000000000000000000) != 0) + res = + (res * 0x18ebef9eac820ae8682b9793ac6d1e778) / + 0x1368b2fc6f9609fe7aceb46aa619baed4; + if ((x & 0x040000000000000000000000000000000) != 0) + res = + (res * 0x1368b2fc6f9609fe7aceb46aa619baed5) / + 0x0bc5ab1b16779be3575bd8f0520a9f21f; + if ((x & 0x080000000000000000000000000000000) != 0) + res = + (res * 0x0bc5ab1b16779be3575bd8f0520a9f21e) / + 0x0454aaa8efe072e7f6ddbab84b40a55c9; + if ((x & 0x100000000000000000000000000000000) != 0) + res = + (res * 0x0454aaa8efe072e7f6ddbab84b40a55c5) / + 0x00960aadc109e7a3bf4578099615711ea; + if ((x & 0x200000000000000000000000000000000) != 0) + res = + (res * 0x00960aadc109e7a3bf4578099615711d7) / + 0x0002bf84208204f5977f9a8cf01fdce3d; + if ((x & 0x400000000000000000000000000000000) != 0) + res = + (res * 0x0002bf84208204f5977f9a8cf01fdc307) / + 0x0000003c6ab775dd0b95b4cbee7e65d11; + + return res; + } + /* solium-enable */ +} diff --git a/contracts/libs/WeightedAverage.sol b/contracts/libs/WeightedAverage.sol index 77712bbf..edf7f363 100644 --- a/contracts/libs/WeightedAverage.sol +++ b/contracts/libs/WeightedAverage.sol @@ -4,34 +4,17 @@ pragma solidity ^0.8.0; library WeightedAverage { uint256 private constant _PRECISION = 10**18; - /* - EXAMPLE: - _PRECISION = 500 - block.timestamp - startTime = 70 - endTime - startTime = 100 - - // scenario 1 : targetAmount > amount - amount = 87 - targetAmount = 137 - - ### pt 1 - ( _PRECISION*amount + _PRECISION * (targetAmount - amount) * 0.7 ) / _PRECISION; - ( 500*87 + 500 * (137 - 87) * 0.7 ) / 500 = 122 - ### pt 2 - ( _PRECISION*amount - _PRECISION * (amount - targetAmount) * 0.7 ) / _PRECISION; - ( 500*87 - 500 * (87 - 137) * 0.7 ) / 500 = 122 - - // scenario 2 : targetAmount < amount - amount = 201 - targetAmount = 172 - - ### pt 1 - ( _PRECISION*amount + _PRECISION * (targetAmount - amount) * 0.7 ) / _PRECISION; - ( 500*201 + 500 * (172 - 201) * 0.7 ) / 500 = 180.7 - ### pt 2 - ( _PRECISION*amount - _PRECISION * (amount - targetAmount) * 0.7 ) / _PRECISION; - ( 500*201 - 500 * (201 - 172) * 0.7 ) / 500 = 180.7 - */ + // CALCULATE TIME-WEIGHTED AVERAGE + /**************************************************************************** + // __ __ // + // wA = weightedAmount / \ // + // a = amout | (a - tA) * (bT - sT) | // + // tA = targetAmount wA = a + | -------------------- | // + // sT = startTime | (eT - sT) | // + // eT = endTime \__ __/ // + // bT = block.timestame // + // // + ****************************************************************************/ function calculate( uint256 amount, @@ -48,6 +31,7 @@ library WeightedAverage { } else { // Currently in an update, return weighted average if (targetAmount > amount) { + // re-orders above visualized formula to handle negative numbers return (_PRECISION * amount + @@ -56,6 +40,7 @@ library WeightedAverage { (block.timestamp - startTime)) / (endTime - startTime)) / _PRECISION; } else { + // follows order of visualized formula above return (_PRECISION * amount - diff --git a/test/contracts/curves/BancorZeroCurve.ts b/test/contracts/curves/BancorZeroCurve.ts index 0260414b..999a3769 100644 --- a/test/contracts/curves/BancorZeroCurve.ts +++ b/test/contracts/curves/BancorZeroCurve.ts @@ -1,7 +1,7 @@ import { SignerWithAddress } from "@nomiclabs/hardhat-ethers/signers"; import { BigNumber, Signer } from "ethers"; import { ethers, getNamedAccounts } from "hardhat"; -import { BancorZeroCurve } from "../../../artifacts/types/BancorZeroCurve"; +import { BancorBancor } from "../../../artifacts/types/BancorBancor"; import { CurveRegistry } from "../../../artifacts/types/CurveRegistry"; import { ERC20 } from "../../../artifacts/types/ERC20"; import { Foundry } from "../../../artifacts/types/Foundry"; @@ -21,7 +21,7 @@ describe("BancorZeroCurve", () => { let weightedAverage: WeightedAverage; let meTokenRegistry: MeTokenRegistry; let meTokenFactory: MeTokenFactory; - let bancorZeroCurve: BancorZeroCurve; + let bancorZeroCurve: BancorBancor; let curveRegistry: CurveRegistry; let vaultRegistry: VaultRegistry; let migrationRegistry: MigrationRegistry; @@ -53,7 +53,7 @@ describe("BancorZeroCurve", () => { ["address"], [DAI] ); - bancorZeroCurve = await deploy("BancorZeroCurve"); + bancorZeroCurve = await deploy("BancorBancor"); let token; ({ diff --git a/test/contracts/curves/Curve.ts b/test/contracts/curves/Curve.ts index 1ef6b92d..b058d56b 100644 --- a/test/contracts/curves/Curve.ts +++ b/test/contracts/curves/Curve.ts @@ -3,7 +3,7 @@ import { SignerWithAddress } from "@nomiclabs/hardhat-ethers/signers"; import { deploy, getContractAt } from "../../utils/helpers"; import { BigNumber, Signer } from "ethers"; import { impersonate, mineBlock, passOneHour } from "../../utils/hardhatNode"; -import { BancorZeroCurve } from "../../../artifacts/types/BancorZeroCurve"; +import { BancorBancor } from "../../../artifacts/types/BancorBancor"; import { ERC20 } from "../../../artifacts/types/ERC20"; import { Foundry } from "../../../artifacts/types/Foundry"; import { Hub } from "../../../artifacts/types/Hub"; @@ -21,7 +21,7 @@ describe("Generic Curve", () => { let account0: SignerWithAddress; let account1: SignerWithAddress; let account2: SignerWithAddress; - let _curve: BancorZeroCurve; + let _curve: BancorBancor; let meTokenRegistry: MeTokenRegistry; let foundry: Foundry; let token: ERC20; @@ -43,7 +43,7 @@ describe("Generic Curve", () => { // TODO: then loop over array of set of curve arguments const MAX_WEIGHT = 1000000; const reserveWeight = MAX_WEIGHT / 2; - const baseY = PRECISION.div(1000).toString(); + const baseY = PRECISION.div(10).toString(); before(async () => { ({ DAI, DAIWhale } = await getNamedAccounts()); @@ -56,7 +56,7 @@ describe("Generic Curve", () => { ["uint256", "uint32"], [baseY, reserveWeight] ); - _curve = await deploy("BancorZeroCurve"); + _curve = await deploy("BancorBancor"); ({ token, @@ -87,6 +87,9 @@ describe("Generic Curve", () => { await dai .connect(account1) .approve(meTokenRegistry.address, ethers.utils.parseEther("100")); + await dai + .connect(account1) + .approve(singleAssetVault.address, ethers.utils.parseEther("100")); }); describe("register()", () => { @@ -106,6 +109,210 @@ describe("Generic Curve", () => { it("Returns correct registered details", async () => {}); }); + describe("viewMeTokensMinted()", () => { + it.only("balanceLocked = 0, balancePooled = 0, mint on meToken creation", async () => { + let expectedMeTokensMinted = await _curve.viewMeTokensMinted( + amount1, + hubId, + 0, + 0 + ); + console.log( + `expectedMeTokensMinted:${ethers.utils.formatEther( + expectedMeTokensMinted + )}` + ); + + // Get balances before mint + let minterDaiBalanceBefore = await dai.balanceOf(account1.address); + console.log( + `minterDaiBalanceBefore:${ethers.utils.formatEther( + minterDaiBalanceBefore + )}` + ); + let vaultDaiBalanceBefore = await dai.balanceOf(singleAssetVault.address); + console.log( + `vaultDaiBalanceBefore:${ethers.utils.formatEther( + vaultDaiBalanceBefore + )}` + ); + + // Mint first meTokens to owner + let tx = await meTokenRegistry + .connect(account1) + .subscribe(name, symbol, hubId, amount1); + let meTokenAddr = await meTokenRegistry.getOwnerMeToken(account1.address); + meToken = await getContractAt("MeToken", meTokenAddr); + + let expectedAssetsDeposited = await _curve.viewAssetsDeposited( + expectedMeTokensMinted, + hubId, + 0, + 0 + ); + console.log( + `expectedAssetsDeposited:${ethers.utils.formatEther( + expectedAssetsDeposited + )}` + ); + + // Compare expected meTokens minted to actual held + let meTokensMinted = await meToken.balanceOf(account1.address); + expect(meTokensMinted).to.equal(expectedMeTokensMinted); + let totalSupply = await meToken.totalSupply(); + console.log(`totalSupply: ${ethers.utils.formatEther(totalSupply)}`); + expect(totalSupply).to.equal(meTokensMinted); + + // Compare owner dai balance before/after + let minterDaiBalanceAfter = await dai.balanceOf(account1.address); + console.log( + `minterDaiBalanceAfter:${ethers.utils.formatEther( + minterDaiBalanceAfter + )}` + ); + expect( + // TODO: how to verify difference of numbers to type of amount1? + minterDaiBalanceBefore.sub(minterDaiBalanceAfter) + ).to.equal(amount1); + + // Expect balance of vault to have increased by assets deposited + let vaultDaiBalanceAfter = await dai.balanceOf(singleAssetVault.address); + console.log( + `vaultDaiBalanceAfterMint:${ethers.utils.formatEther( + vaultDaiBalanceAfter + )}` + ); + + // Burn meTokens to owner + let assetsReturned = await _curve.viewAssetsReturned( + meTokensMinted.div(2), + hubId, + totalSupply, + amount1 + ); + + console.log( + `viewAssetsReturned: ${ethers.utils.formatEther(assetsReturned)}` + ); + + await foundry + .connect(account1) + .burn( + meToken.address, + meTokensMinted.div(BigNumber.from(2)), + account1.address + ); + console.log( + `meTokens burned: ${ethers.utils.formatEther( + meTokensMinted.div(BigNumber.from(2)) + )}` + ); + + let newSupply = await meToken.totalSupply(); + let newDaiBalance = await dai.balanceOf(account1.address); + console.log(`Supply after burn: ${ethers.utils.formatEther(newSupply)}`); + console.log( + `Owners' DAI balance after burn: ${ethers.utils.formatEther( + newDaiBalance + )}` + ); + + expect(vaultDaiBalanceAfter.sub(vaultDaiBalanceBefore)).to.equal(amount1); + expect(amount1).to.equal(expectedAssetsDeposited); + }); + + it("balanceLocked = 0, balancePooled = 0, mint after meToken creation", async () => { + let expectedMeTokensMinted = await _curve.viewMeTokensMinted( + amount1, + hubId, + 0, + 0 + ); + let expectedAssetsDeposited = await _curve.viewAssetsDeposited( + expectedMeTokensMinted, + hubId, + 0, + 0 + ); + + // Get balances before mint + let minterDaiBalanceBefore = await dai.balanceOf(account2.address); + let vaultDaiBalanceBefore = await dai.balanceOf(singleAssetVault.address); + + // Create meToken w/o issuing supply + const tx = await meTokenRegistry + .connect(account2) + .subscribe(name, symbol, hubId, 0); + const meTokenAddr = await meTokenRegistry.getOwnerMeToken( + account2.address + ); + meToken = await getContractAt("MeToken", meTokenAddr); + + // Mint meToken + await foundry + .connect(account2) + .mint(meToken.address, amount1, account2.address); + + // Compare expected meTokens minted to actual held + const meTokensMinted = await meToken.balanceOf(account2.address); + expect(meTokensMinted).to.equal(expectedMeTokensMinted); + const totalSupply = await meToken.totalSupply(); + expect(totalSupply).to.equal(meTokensMinted); + + // Compare buyer dai balance before/after + let minterDaiBalanceAfter = await dai.balanceOf(account2.address); + expect( + Number(minterDaiBalanceBefore) - Number(minterDaiBalanceAfter) + ).to.equal(amount1); + + // Expect balance of vault to have increased by assets deposited + let vaultDaiBalanceAfter = await dai.balanceOf(singleAssetVault.address); + expect( + Number(vaultDaiBalanceAfter) - Number(vaultDaiBalanceBefore) + ).to.equal(amount1); + expect(amount1).to.equal(expectedAssetsDeposited); + }); + + it("balanceLocked = 0, balancePooled > 0", async () => { + // TODO + }); + + it("balanceLocked > 0, balancePooled = 0", async () => { + // TODO + }); + + it("balanceLocked > 0, balancePooled > 0", async () => { + // TODO + }); + }); + + describe("calculateBurnReturn()", () => { + it("balanceLocked = 0, buyer, ending supply = 0", async () => { + // TODO + }); + it("balanceLocked = 0, owner, ending supply = 0", async () => { + // TODO + }); + it("balanceLocked = 0, buyer, ending supply > 0", async () => { + // TODO + }); + it("balanceLocked = 0, owner, ending supply > 0", async () => { + // TODO + }); + it("balanceLocked > 0, buyer, ending supply = 0", async () => { + // TODO + }); + it("balanceLocked > 0, owner, ending supply = 0", async () => { + // TODO + }); + it("balanceLocked > 0, buyer, ending supply > 0", async () => { + // TODO + }); + it("balanceLocked > 0, owner, ending supply > 0", async () => { + // TODO + }); + }); + describe("initReconfigure()", () => { it("Only be callable by Hub", async () => { // TODO