/*
Copyright 2020 dYdX Trading Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
pragma solidity 0.5.16;
pragma experimental ABIEncoderV2;
// File: @openzeppelin/contracts/math/SafeMath.sol
/**
* @dev Wrappers over Solidity's arithmetic operations with added overflow
* checks.
*
* Arithmetic operations in Solidity wrap on overflow. This can easily result
* in bugs, because programmers usually assume that an overflow raises an
* error, which is the standard behavior in high level programming languages.
* `SafeMath` restores this intuition by reverting the transaction when an
* operation overflows.
*
* Using this library instead of the unchecked operations eliminates an entire
* class of bugs, so it's recommended to use it always.
*/
library SafeMath {
/**
* @dev Returns the addition of two unsigned integers, reverting on
* overflow.
*
* Counterpart to Solidity's `+` operator.
*
* Requirements:
* - Addition cannot overflow.
*/
function add(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
require(c >= a, "SafeMath: addition overflow");
return c;
}
/**
* @dev Returns the subtraction of two unsigned integers, reverting on
* overflow (when the result is negative).
*
* Counterpart to Solidity's `-` operator.
*
* Requirements:
* - Subtraction cannot overflow.
*/
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
return sub(a, b, "SafeMath: subtraction overflow");
}
/**
* @dev Returns the subtraction of two unsigned integers, reverting with custom message on
* overflow (when the result is negative).
*
* Counterpart to Solidity's `-` operator.
*
* Requirements:
* - Subtraction cannot overflow.
*
* _Available since v2.4.0._
*/
function sub(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
require(b <= a, errorMessage);
uint256 c = a - b;
return c;
}
/**
* @dev Returns the multiplication of two unsigned integers, reverting on
* overflow.
*
* Counterpart to Solidity's `*` operator.
*
* Requirements:
* - Multiplication cannot overflow.
*/
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
// Gas optimization: this is cheaper than requiring 'a' not being zero, but the
// benefit is lost if 'b' is also tested.
// See: https://github.com/OpenZeppelin/openzeppelin-contracts/pull/522
if (a == 0) {
return 0;
}
uint256 c = a * b;
require(c / a == b, "SafeMath: multiplication overflow");
return c;
}
/**
* @dev Returns the integer division of two unsigned integers. Reverts on
* division by zero. The result is rounded towards zero.
*
* Counterpart to Solidity's `/` operator. Note: this function uses a
* `revert` opcode (which leaves remaining gas untouched) while Solidity
* uses an invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/
function div(uint256 a, uint256 b) internal pure returns (uint256) {
return div(a, b, "SafeMath: division by zero");
}
/**
* @dev Returns the integer division of two unsigned integers. Reverts with custom message on
* division by zero. The result is rounded towards zero.
*
* Counterpart to Solidity's `/` operator. Note: this function uses a
* `revert` opcode (which leaves remaining gas untouched) while Solidity
* uses an invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*
* _Available since v2.4.0._
*/
function div(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
// Solidity only automatically asserts when dividing by 0
require(b > 0, errorMessage);
uint256 c = a / b;
// assert(a == b * c + a % b); // There is no case in which this doesn't hold
return c;
}
/**
* @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
* Reverts when dividing by zero.
*
* Counterpart to Solidity's `%` operator. This function uses a `revert`
* opcode (which leaves remaining gas untouched) while Solidity uses an
* invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*/
function mod(uint256 a, uint256 b) internal pure returns (uint256) {
return mod(a, b, "SafeMath: modulo by zero");
}
/**
* @dev Returns the remainder of dividing two unsigned integers. (unsigned integer modulo),
* Reverts with custom message when dividing by zero.
*
* Counterpart to Solidity's `%` operator. This function uses a `revert`
* opcode (which leaves remaining gas untouched) while Solidity uses an
* invalid opcode to revert (consuming all remaining gas).
*
* Requirements:
* - The divisor cannot be zero.
*
* _Available since v2.4.0._
*/
function mod(uint256 a, uint256 b, string memory errorMessage) internal pure returns (uint256) {
require(b != 0, errorMessage);
return a % b;
}
}
// File: @openzeppelin/contracts/GSN/Context.sol
/*
* @dev Provides information about the current execution context, including the
* sender of the transaction and its data. While these are generally available
* via msg.sender and msg.data, they should not be accessed in such a direct
* manner, since when dealing with GSN meta-transactions the account sending and
* paying for execution may not be the actual sender (as far as an application
* is concerned).
*
* This contract is only required for intermediate, library-like contracts.
*/
contract Context {
// Empty internal constructor, to prevent people from mistakenly deploying
// an instance of this contract, which should be used via inheritance.
constructor () internal { }
// solhint-disable-previous-line no-empty-blocks
function _msgSender() internal view returns (address payable) {
return msg.sender;
}
function _msgData() internal view returns (bytes memory) {
this; // silence state mutability warning without generating bytecode - see https://github.com/ethereum/solidity/issues/2691
return msg.data;
}
}
// File: @openzeppelin/contracts/ownership/Ownable.sol
/**
* @dev Contract module which provides a basic access control mechanism, where
* there is an account (an owner) that can be granted exclusive access to
* specific functions.
*
* This module is used through inheritance. It will make available the modifier
* `onlyOwner`, which can be applied to your functions to restrict their use to
* the owner.
*/
contract Ownable is Context {
address private _owner;
event OwnershipTransferred(address indexed previousOwner, address indexed newOwner);
/**
* @dev Initializes the contract setting the deployer as the initial owner.
*/
constructor () internal {
address msgSender = _msgSender();
_owner = msgSender;
emit OwnershipTransferred(address(0), msgSender);
}
/**
* @dev Returns the address of the current owner.
*/
function owner() public view returns (address) {
return _owner;
}
/**
* @dev Throws if called by any account other than the owner.
*/
modifier onlyOwner() {
require(isOwner(), "Ownable: caller is not the owner");
_;
}
/**
* @dev Returns true if the caller is the current owner.
*/
function isOwner() public view returns (bool) {
return _msgSender() == _owner;
}
/**
* @dev Leaves the contract without owner. It will not be possible to call
* `onlyOwner` functions anymore. Can only be called by the current owner.
*
* NOTE: Renouncing ownership will leave the contract without an owner,
* thereby removing any functionality that is only available to the owner.
*/
function renounceOwnership() public onlyOwner {
emit OwnershipTransferred(_owner, address(0));
_owner = address(0);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
* Can only be called by the current owner.
*/
function transferOwnership(address newOwner) public onlyOwner {
_transferOwnership(newOwner);
}
/**
* @dev Transfers ownership of the contract to a new account (`newOwner`).
*/
function _transferOwnership(address newOwner) internal {
require(newOwner != address(0), "Ownable: new owner is the zero address");
emit OwnershipTransferred(_owner, newOwner);
_owner = newOwner;
}
}
// File: contracts/protocol/lib/BaseMath.sol
/**
* @title BaseMath
* @author dYdX
*
* @dev Arithmetic for fixed-point numbers with 18 decimals of precision.
*/
library BaseMath {
using SafeMath for uint256;
// The number One in the BaseMath system.
uint256 constant internal BASE = 10 ** 18;
/**
* @dev Getter function since constants can't be read directly from libraries.
*/
function base()
internal
pure
returns (uint256)
{
return BASE;
}
/**
* @dev Multiplies a value by a base value (result is rounded down).
*/
function baseMul(
uint256 value,
uint256 baseValue
)
internal
pure
returns (uint256)
{
return value.mul(baseValue).div(BASE);
}
/**
* @dev Multiplies a value by a base value (result is rounded down).
* Intended as an alternaltive to baseMul to prevent overflow, when `value` is known
* to be divisible by `BASE`.
*/
function baseDivMul(
uint256 value,
uint256 baseValue
)
internal
pure
returns (uint256)
{
return value.div(BASE).mul(baseValue);
}
/**
* @dev Multiplies a value by a base value (result is rounded up).
*/
function baseMulRoundUp(
uint256 value,
uint256 baseValue
)
internal
pure
returns (uint256)
{
if (value == 0 || baseValue == 0) {
return 0;
}
return value.mul(baseValue).sub(1).div(BASE).add(1);
}
/**
* @dev Divide a value by a base value (result is rounded down).
*/
function baseDiv(
uint256 value,
uint256 baseValue
)
internal
pure
returns (uint256)
{
return value.mul(BASE).div(baseValue);
}
/**
* @dev Returns a base value representing the reciprocal of another base value (result is
* rounded down).
*/
function baseReciprocal(
uint256 baseValue
)
internal
pure
returns (uint256)
{
return baseDiv(BASE, baseValue);
}
}
// File: contracts/protocol/lib/Math.sol
/**
* @title Math
* @author dYdX
*
* @dev Library for non-standard Math functions.
*/
library Math {
using SafeMath for uint256;
// ============ Library Functions ============
/**
* @dev Return target * (numerator / denominator), rounded down.
*/
function getFraction(
uint256 target,
uint256 numerator,
uint256 denominator
)
internal
pure
returns (uint256)
{
return target.mul(numerator).div(denominator);
}
/**
* @dev Return target * (numerator / denominator), rounded up.
*/
function getFractionRoundUp(
uint256 target,
uint256 numerator,
uint256 denominator
)
internal
pure
returns (uint256)
{
if (target == 0 || numerator == 0) {
// SafeMath will check for zero denominator
return SafeMath.div(0, denominator);
}
return target.mul(numerator).sub(1).div(denominator).add(1);
}
/**
* @dev Returns the minimum between a and b.
*/
function min(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
return a < b ? a : b;
}
/**
* @dev Returns the maximum between a and b.
*/
function max(
uint256 a,
uint256 b
)
internal
pure
returns (uint256)
{
return a > b ? a : b;
}
}
// File: contracts/protocol/lib/SafeCast.sol
/**
* @title SafeCast
* @author dYdX
*
* @dev Library for casting uint256 to other types of uint.
*/
library SafeCast {
/**
* @dev Returns the downcasted uint128 from uint256, reverting on
* overflow (i.e. when the input is greater than largest uint128).
*
* Counterpart to Solidity's `uint128` operator.
*
* Requirements:
* - `value` must fit into 128 bits.
*/
function toUint128(
uint256 value
)
internal
pure
returns (uint128)
{
require(value < 2**128, "SafeCast: value doesn\'t fit in 128 bits");
return uint128(value);
}
/**
* @dev Returns the downcasted uint120 from uint256, reverting on
* overflow (i.e. when the input is greater than largest uint120).
*
* Counterpart to Solidity's `uint120` operator.
*
* Requirements:
* - `value` must fit into 120 bits.
*/
function toUint120(
uint256 value
)
internal
pure
returns (uint120)
{
require(value < 2**120, "SafeCast: value doesn\'t fit in 120 bits");
return uint120(value);
}
/**
* @dev Returns the downcasted uint32 from uint256, reverting on
* overflow (i.e. when the input is greater than largest uint32).
*
* Counterpart to Solidity's `uint32` operator.
*
* Requirements:
* - `value` must fit into 32 bits.
*/
function toUint32(
uint256 value
)
internal
pure
returns (uint32)
{
require(value < 2**32, "SafeCast: value doesn\'t fit in 32 bits");
return uint32(value);
}
}
// File: contracts/protocol/lib/SignedMath.sol
/**
* @title SignedMath
* @author dYdX
*
* @dev SignedMath library for doing math with signed integers.
*/
library SignedMath {
using SafeMath for uint256;
// ============ Structs ============
struct Int {
uint256 value;
bool isPositive;
}
// ============ Functions ============
/**
* @dev Returns a new signed integer equal to a signed integer plus an unsigned integer.
*/
function add(
Int memory sint,
uint256 value
)
internal
pure
returns (Int memory)
{
if (sint.isPositive) {
return Int({
value: value.add(sint.value),
isPositive: true
});
}
if (sint.value < value) {
return Int({
value: value.sub(sint.value),
isPositive: true
});
}
return Int({
value: sint.value.sub(value),
isPositive: false
});
}
/**
* @dev Returns a new signed integer equal to a signed integer minus an unsigned integer.
*/
function sub(
Int memory sint,
uint256 value
)
internal
pure
returns (Int memory)
{
if (!sint.isPositive) {
return Int({
value: value.add(sint.value),
isPositive: false
});
}
if (sint.value > value) {
return Int({
value: sint.value.sub(value),
isPositive: true
});
}
return Int({
value: value.sub(sint.value),
isPositive: false
});
}
/**
* @dev Returns a new signed integer equal to a signed integer plus another signed integer.
*/
function signedAdd(
Int memory augend,
Int memory addend
)
internal
pure
returns (Int memory)
{
return addend.isPositive
? add(augend, addend.value)
: sub(augend, addend.value);
}
/**
* @dev Returns a new signed integer equal to a signed integer minus another signed integer.
*/
function signedSub(
Int memory minuend,
Int memory subtrahend
)
internal
pure
returns (Int memory)
{
return subtrahend.isPositive
? sub(minuend, subtrahend.value)
: add(minuend, subtrahend.value);
}
/**
* @dev Returns true if signed integer `a` is greater than signed integer `b`, false otherwise.
*/
function gt(
Int memory a,
Int memory b
)
internal
pure
returns (bool)
{
if (a.isPositive) {
if (b.isPositive) {
return a.value > b.value;
} else {
// True, unless both values are zero.
return a.value != 0 || b.value != 0;
}
} else {
if (b.isPositive) {
return false;
} else {
return a.value < b.value;
}
}
}
/**
* @dev Returns the minimum of signed integers `a` and `b`.
*/
function min(
Int memory a,
Int memory b
)
internal
pure
returns (Int memory)
{
return gt(b, a) ? a : b;
}
/**
* @dev Returns the maximum of signed integers `a` and `b`.
*/
function max(
Int memory a,
Int memory b
)
internal
pure
returns (Int memory)
{
return gt(a, b) ? a : b;
}
}
// File: contracts/protocol/v1/intf/I_P1Funder.sol
/**
* @title I_P1Funder
* @author dYdX
*
* @notice Interface for an oracle providing the funding rate for a perpetual market.
*/
interface I_P1Funder {
/**
* @notice Calculates the signed funding amount that has accumulated over a period of time.
*
* @param timeDelta Number of seconds over which to calculate the accumulated funding amount.
* @return True if the funding rate is positive, and false otherwise.
* @return The funding amount as a unitless rate, represented as a fixed-point number
* with 18 decimals.
*/
function getFunding(
uint256 timeDelta
)
external
view
returns (bool, uint256);
}
// File: contracts/protocol/v1/lib/P1Types.sol
/**
* @title P1Types
* @author dYdX
*
* @dev Library for common types used in PerpetualV1 contracts.
*/
library P1Types {
// ============ Structs ============
/**
* @dev Used to represent the global index and each account's cached index.
* Used to settle funding paymennts on a per-account basis.
*/
struct Index {
uint32 timestamp;
bool isPositive;
uint128 value;
}
/**
* @dev Used to track the signed margin balance and position balance values for each account.
*/
struct Balance {
bool marginIsPositive;
bool positionIsPositive;
uint120 margin;
uint120 position;
}
/**
* @dev Used to cache commonly-used variables that are relatively gas-intensive to obtain.
*/
struct Context {
uint256 price;
uint256 minCollateral;
Index index;
}
/**
* @dev Used by contracts implementing the I_P1Trader interface to return the result of a trade.
*/
struct TradeResult {
uint256 marginAmount;
uint256 positionAmount;
bool isBuy; // From taker's perspective.
bytes32 traderFlags;
}
}
// File: contracts/protocol/v1/lib/P1IndexMath.sol
/**
* @title P1IndexMath
* @author dYdX
*
* @dev Library for manipulating P1Types.Index structs.
*/
library P1IndexMath {
// ============ Constants ============
uint256 private constant FLAG_IS_POSITIVE = 1 << (8 * 16);
// ============ Functions ============
/**
* @dev Returns a compressed bytes32 representation of the index for logging.
*/
function toBytes32(
P1Types.Index memory index
)
internal
pure
returns (bytes32)
{
uint256 result =
index.value
| (index.isPositive ? FLAG_IS_POSITIVE : 0)
| (uint256(index.timestamp) << 136);
return bytes32(result);
}
}
// File: contracts/protocol/v1/oracles/P1FundingOracle.sol
/**
* @title P1FundingOracle
* @author dYdX
*
* @notice Oracle providing the funding rate for a perpetual market.
*/
contract P1FundingOracle is
Ownable,
I_P1Funder
{
using BaseMath for uint256;
using SafeCast for uint256;
using SafeMath for uint128;
using SafeMath for uint256;
using P1IndexMath for P1Types.Index;
using SignedMath for SignedMath.Int;
// ============ Constants ============
uint256 private constant FLAG_IS_POSITIVE = 1 << 128;
uint128 constant internal BASE = 10 ** 18;
/**
* @notice Bounding params constraining updates to the funding rate.
*
* Like the funding rate, these are per-second rates, fixed-point with 18 decimals.
* We calculate the per-second rates from the market specifications, which use 8-hour rates:
* - The max absolute funding rate is 0.75% (8-hour rate).
* - The max change over a 45-minute period is 1.5% (8-hour rate).
*
* This means the fastest the funding rate can go from its min to its max value, or vice versa,
* is in 45 minutes.
*/
uint128 public constant MAX_ABS_VALUE = BASE * 75 / 10000 / (8 hours);
uint128 public constant MAX_ABS_DIFF_PER_SECOND = MAX_ABS_VALUE * 2 / (45 minutes);
// ============ Events ============
event LogFundingRateUpdated(
bytes32 fundingRate
);
event LogFundingRateProviderSet(
address fundingRateProvider
);
// ============ Mutable Storage ============
// The funding rate is denoted in units per second, as a fixed-point number with 18 decimals.
P1Types.Index private _FUNDING_RATE_;
// Address which has the ability to update the funding rate.
address public _FUNDING_RATE_PROVIDER_;
// ============ Constructor ============
constructor(
address fundingRateProvider
)
public
{
P1Types.Index memory fundingRate = P1Types.Index({
timestamp: block.timestamp.toUint32(),
isPositive: true,
value: 0
});
_FUNDING_RATE_ = fundingRate;
_FUNDING_RATE_PROVIDER_ = fundingRateProvider;
emit LogFundingRateUpdated(fundingRate.toBytes32());
emit LogFundingRateProviderSet(fundingRateProvider);
}
// ============ External Functions ============
/**
* @notice Set the funding rate, denoted in units per second, fixed-point with 18 decimals.
* @dev Can only be called by the funding rate provider. Emits the LogFundingRateUpdated event.
*
* @param newRate The intended new funding rate. Is bounded by the global constant bounds.
* @return The new funding rate with a timestamp of the update.
*/
function setFundingRate(
SignedMath.Int calldata newRate
)
external
returns (P1Types.Index memory)
{
require(
msg.sender == _FUNDING_RATE_PROVIDER_,
"The funding rate can only be set by the funding rate provider"
);
SignedMath.Int memory boundedNewRate = _boundRate(newRate);
P1Types.Index memory boundedNewRateWithTimestamp = P1Types.Index({
timestamp: block.timestamp.toUint32(),
isPositive: boundedNewRate.isPositive,
value: boundedNewRate.value.toUint128()
});
_FUNDING_RATE_ = boundedNewRateWithTimestamp;
emit LogFundingRateUpdated(boundedNewRateWithTimestamp.toBytes32());
return boundedNewRateWithTimestamp;
}
/**
* @notice Set the funding rate provider. Can only be called by the admin.
* @dev Emits the LogFundingRateProviderSet event.
*
* @param newProvider The new provider, who will have the ability to set the funding rate.
*/
function setFundingRateProvider(
address newProvider
)
external
onlyOwner
{
_FUNDING_RATE_PROVIDER_ = newProvider;
emit LogFundingRateProviderSet(newProvider);
}
// ============ Public Functions ============
/**
* @notice Calculates the signed funding amount that has accumulated over a period of time.
*
* @param timeDelta Number of seconds over which to calculate the accumulated funding amount.
* @return True if the funding rate is positive, and false otherwise.
* @return The funding amount as a unitless rate, represented as a fixed-point number
* with 18 decimals.
*/
function getFunding(
uint256 timeDelta
)
public
view
returns (bool, uint256)
{
// Note: Funding interest in PerpetualV1 does not compound, as the interest affects margin
// balances but is calculated based on position balances.
P1Types.Index memory fundingRate = _FUNDING_RATE_;
uint256 fundingAmount = uint256(fundingRate.value).mul(timeDelta);
return (fundingRate.isPositive, fundingAmount);
}
// ============ Helper Functions ============
/**
* @dev Apply the contract-defined bounds and return the bounded rate.
*/
function _boundRate(
SignedMath.Int memory newRate
)
private
view
returns (SignedMath.Int memory)
{
// Get the old rate from storage.
P1Types.Index memory oldRateWithTimestamp = _FUNDING_RATE_;
SignedMath.Int memory oldRate = SignedMath.Int({
value: oldRateWithTimestamp.value,
isPositive: oldRateWithTimestamp.isPositive
});
// Get the maximum allowed change in the rate.
uint256 timeDelta = block.timestamp.sub(oldRateWithTimestamp.timestamp);
uint256 maxDiff = MAX_ABS_DIFF_PER_SECOND.mul(timeDelta);
// Calculate and return the bounded rate.
if (newRate.gt(oldRate)) {
SignedMath.Int memory upperBound = SignedMath.min(
oldRate.add(maxDiff),
SignedMath.Int({ value: MAX_ABS_VALUE, isPositive: true })
);
return SignedMath.min(
newRate,
upperBound
);
} else {
SignedMath.Int memory lowerBound = SignedMath.max(
oldRate.sub(maxDiff),
SignedMath.Int({ value: MAX_ABS_VALUE, isPositive: false })
);
return SignedMath.max(
newRate,
lowerBound
);
}
}
}
// File: contracts/protocol/v1/oracles/P1InverseFundingOracle.sol
/**
* @title P1InverseFundingOracle
* @author dYdX
*
* @notice P1FundingOracle that uses the inverted rate (i.e. flips base and quote currencies)
* when getting the funding amount.
*/
contract P1InverseFundingOracle is
P1FundingOracle
{
// ============ Constructor ============
constructor(
address fundingRateProvider
)
P1FundingOracle(fundingRateProvider)
public
{
}
// ============ External Functions ============
/**
* @notice Calculates the signed funding amount that has accumulated over a period of time.
*
* @param timeDelta Number of seconds over which to calculate the accumulated funding amount.
* @return True if the funding rate is positive, and false otherwise.
* @return The funding amount as a unitless rate, represented as a fixed-point number
* with 18 decimals.
*/
function getFunding(
uint256 timeDelta
)
public
view
returns (bool, uint256)
{
(bool isPositive, uint256 fundingAmount) = super.getFunding(timeDelta);
return (!isPositive, fundingAmount);
}
}
{
"compilationTarget": {
"P1InverseFundingOracle.sol": "P1InverseFundingOracle"
},
"evmVersion": "istanbul",
"libraries": {},
"optimizer": {
"enabled": true,
"runs": 10000
},
"remappings": []
}
[{"inputs":[{"internalType":"address","name":"fundingRateProvider","type":"address"}],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"address","name":"fundingRateProvider","type":"address"}],"name":"LogFundingRateProviderSet","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"internalType":"bytes32","name":"fundingRate","type":"bytes32"}],"name":"LogFundingRateUpdated","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"internalType":"address","name":"previousOwner","type":"address"},{"indexed":true,"internalType":"address","name":"newOwner","type":"address"}],"name":"OwnershipTransferred","type":"event"},{"constant":true,"inputs":[],"name":"MAX_ABS_DIFF_PER_SECOND","outputs":[{"internalType":"uint128","name":"","type":"uint128"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"MAX_ABS_VALUE","outputs":[{"internalType":"uint128","name":"","type":"uint128"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"_FUNDING_RATE_PROVIDER_","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"internalType":"uint256","name":"timeDelta","type":"uint256"}],"name":"getFunding","outputs":[{"internalType":"bool","name":"","type":"bool"},{"internalType":"uint256","name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"isOwner","outputs":[{"internalType":"bool","name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"owner","outputs":[{"internalType":"address","name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"renounceOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"components":[{"internalType":"uint256","name":"value","type":"uint256"},{"internalType":"bool","name":"isPositive","type":"bool"}],"internalType":"struct SignedMath.Int","name":"newRate","type":"tuple"}],"name":"setFundingRate","outputs":[{"components":[{"internalType":"uint32","name":"timestamp","type":"uint32"},{"internalType":"bool","name":"isPositive","type":"bool"},{"internalType":"uint128","name":"value","type":"uint128"}],"internalType":"struct P1Types.Index","name":"","type":"tuple"}],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"newProvider","type":"address"}],"name":"setFundingRateProvider","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"internalType":"address","name":"newOwner","type":"address"}],"name":"transferOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"}]