-
Notifications
You must be signed in to change notification settings - Fork 19
/
Copy pathMarket.sol
616 lines (521 loc) · 26.6 KB
/
Market.sol
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
// SPDX-License-Identifier: Apache-2.0
pragma solidity 0.8.19;
import "@equilibria/root/attribute/Instance.sol";
import "@equilibria/root/attribute/ReentrancyGuard.sol";
import "./interfaces/IMarket.sol";
import "./interfaces/IMarketFactory.sol";
/// @title Market
/// @notice Manages logic and state for a single market.
/// @dev Cloned by the Factory contract to launch new markets.
contract Market is IMarket, Instance, ReentrancyGuard {
/// @dev The underlying token that the market settles in
Token18 public token;
/// @dev The token that incentive rewards are paid in
Token18 public reward;
/// @dev The oracle that provides the market price
IOracleProvider public oracle;
/// @dev The payoff function over the underlying oracle
IPayoffProvider public payoff;
/// @dev Beneficiary of the market, receives donations
address public beneficiary;
/// @dev Risk coordinator of the market
address public coordinator;
/// @dev Risk parameters of the market
RiskParameterStorage private _riskParameter;
/// @dev Parameters of the market
MarketParameterStorage private _parameter;
/// @dev Current global state of the market
GlobalStorage private _global;
/// @dev Current global position of the market
PositionStorageGlobal private _position;
/// @dev The global pending versions for each id
mapping(uint256 => PositionStorageGlobal) private _pendingPosition;
/// @dev Current local state of each account
mapping(address => LocalStorage) private _locals;
/// @dev Current local position of each account
mapping(address => PositionStorageLocal) private _positions;
/// @dev The local pending versions for each id for each account
mapping(address => mapping(uint256 => PositionStorageLocal)) private _pendingPositions;
/// @dev The historical version accumulator data for each accessed version
mapping(uint256 => VersionStorage) private _versions;
/// @notice Initializes the contract state
/// @param definition_ The market definition
function initialize(IMarket.MarketDefinition calldata definition_) external initializer(1) {
__Instance__initialize();
__UReentrancyGuard__initialize();
token = definition_.token;
oracle = definition_.oracle;
payoff = definition_.payoff;
}
/// @notice Updates the account's position and collateral
/// @param account The account to operate on
/// @param newMaker The new maker position for the account
/// @param newMaker The new long position for the account
/// @param newMaker The new short position for the account
/// @param collateral The collateral amount to add or remove from the account
/// @param protect Whether to put the account into a protected status for liquidations
function update(
address account,
UFixed6 newMaker,
UFixed6 newLong,
UFixed6 newShort,
Fixed6 collateral,
bool protect
) external nonReentrant whenNotPaused {
Context memory context = _loadContext(account);
_settle(context, account);
_update(context, account, newMaker, newLong, newShort, collateral, protect);
_saveContext(context, account);
}
/// @notice Updates the beneficiary of the market
/// @param newBeneficiary The new beneficiary address
function updateBeneficiary(address newBeneficiary) external onlyOwner {
beneficiary = newBeneficiary;
emit BeneficiaryUpdated(newBeneficiary);
}
/// @notice Updates the coordinator of the market
/// @param newCoordinator The new coordinator address
function updateCoordinator(address newCoordinator) external onlyOwner {
coordinator = newCoordinator;
emit CoordinatorUpdated(newCoordinator);
}
/// @notice Updates the parameter set of the market
/// @param newParameter The new parameter set
function updateParameter(MarketParameter memory newParameter) external onlyOwner {
_parameter.validateAndStore(newParameter, IMarketFactory(address(factory())).parameter(), reward);
emit ParameterUpdated(newParameter);
}
/// @notice Updates the risk parameter set of the market
/// @param newRiskParameter The new risk parameter set
function updateRiskParameter(RiskParameter memory newRiskParameter) external onlyCoordinator {
_riskParameter.validateAndStore(newRiskParameter, IMarketFactory(address(factory())).parameter());
emit RiskParameterUpdated(newRiskParameter);
}
/// @notice Updates the reward token of the market
/// @param newReward The new reward token
function updateReward(Token18 newReward) public onlyOwner {
if (!reward.eq(Token18Lib.ZERO)) revert MarketRewardAlreadySetError();
if (newReward.eq(token)) revert MarketInvalidRewardError();
reward = newReward;
emit RewardUpdated(newReward);
}
/// @notice Claims any available fee that the sender has accrued
/// @dev Applicable fees include: protocol, oracle, risk, and donation
function claimFee() external {
Global memory newGlobal = _global.read();
if (_claimFee(factory().owner(), newGlobal.protocolFee)) newGlobal.protocolFee = UFixed6Lib.ZERO;
if (_claimFee(address(IMarketFactory(address(factory())).oracleFactory()), newGlobal.oracleFee))
newGlobal.oracleFee = UFixed6Lib.ZERO;
if (_claimFee(coordinator, newGlobal.riskFee)) newGlobal.riskFee = UFixed6Lib.ZERO;
if (_claimFee(beneficiary, newGlobal.donation)) newGlobal.donation = UFixed6Lib.ZERO;
_global.store(newGlobal);
}
/// @notice Helper function to handle a singular fee claim.
/// @param receiver The address to receive the fee
/// @param fee The amount of the fee to claim
function _claimFee(address receiver, UFixed6 fee) private returns (bool) {
if (msg.sender != receiver) return false;
token.push(receiver, UFixed18Lib.from(fee));
emit FeeClaimed(receiver, fee);
return true;
}
/// @notice Claims any available reward that the sender has accrued
function claimReward() external {
Local memory newLocal = _locals[msg.sender].read();
reward.push(msg.sender, UFixed18Lib.from(newLocal.reward));
emit RewardClaimed(msg.sender, newLocal.reward);
newLocal.reward = UFixed6Lib.ZERO;
_locals[msg.sender].store(newLocal);
}
/// @notice Returns the current parameter set
function parameter() external view returns (MarketParameter memory) {
return _parameter.read();
}
/// @notice Returns the current risk parameter set
function riskParameter() external view returns (RiskParameter memory) {
return _riskParameter.read();
}
/// @notice Returns the current global position
function position() external view returns (Position memory) {
return _position.read();
}
/// @notice Returns the current local position for the account
/// @param account The account to query
function positions(address account) external view returns (Position memory) {
return _positions[account].read();
}
/// @notice Returns the current global state
function global() external view returns (Global memory) {
return _global.read();
}
/// @notice Returns the historical version snapshot at the given timestamp
/// @param timestamp The timestamp to query
function versions(uint256 timestamp) external view returns (Version memory) {
return _versions[timestamp].read();
}
/// @notice Returns the local state for the given account
/// @param account The account to query
function locals(address account) external view returns (Local memory) {
return _locals[account].read();
}
/// @notice Returns the global pending position for the given id
/// @param id The id to query
function pendingPosition(uint256 id) external view returns (Position memory) {
return _pendingPosition[id].read();
}
/// @notice Returns the local pending position for the given account and id
/// @param account The account to query
/// @param id The id to query
function pendingPositions(address account, uint256 id) external view returns (Position memory) {
return _pendingPositions[account][id].read();
}
/// @notice Loads the current position context for the given account
/// @param context The context to load to
/// @param account The account to query
function _loadCurrentPositionContext(
Context memory context,
address account
) private view returns (PositionContext memory positionContext) {
positionContext.global = _pendingPosition[context.global.currentId].read();
positionContext.local = _pendingPositions[account][context.local.currentId].read();
if (context.global.currentId == context.global.latestId)
positionContext.global.invalidate(context.latestPosition.global);
if (context.local.currentId == context.local.latestId)
positionContext.local.invalidate(context.latestPosition.local);
}
/// @notice Updates the current position
/// @param context The context to use
/// @param account The account to update
/// @param newMaker The new maker position size
/// @param newLong The new long position size
/// @param newShort The new short position size
/// @param collateral The change in collateral
/// @param protect Whether to protect the position for liquidation
function _update(
Context memory context,
address account,
UFixed6 newMaker,
UFixed6 newLong,
UFixed6 newShort,
Fixed6 collateral,
bool protect
) private {
// read
context.currentPosition = _loadCurrentPositionContext(context, account);
// magic values
if (collateral.eq(Fixed6Lib.MIN)) collateral = context.local.collateral.mul(Fixed6Lib.NEG_ONE);
if (newMaker.eq(UFixed6Lib.MAX)) newMaker = context.currentPosition.local.maker;
if (newLong.eq(UFixed6Lib.MAX)) newLong = context.currentPosition.local.long;
if (newShort.eq(UFixed6Lib.MAX)) newShort = context.currentPosition.local.short;
// advance to next id if applicable
if (context.currentTimestamp > context.currentPosition.local.timestamp) {
context.local.currentId++;
context.currentPosition.local.prepare();
}
if (context.currentTimestamp > context.currentPosition.global.timestamp) {
context.global.currentId++;
context.currentPosition.global.prepare();
}
// update position
Order memory newOrder =
context.currentPosition.local.update(context.currentTimestamp, newMaker, newLong, newShort);
context.currentPosition.global.update(context.currentTimestamp, newOrder, context.riskParameter);
// update fee
newOrder.registerFee(context.latestVersion, context.marketParameter, context.riskParameter);
context.currentPosition.local.registerFee(newOrder);
context.currentPosition.global.registerFee(newOrder);
// update collateral
context.local.update(collateral);
context.currentPosition.local.update(collateral);
// protect account
bool protected = context.local.protect(context.latestPosition.local, context.currentTimestamp, protect);
// request version
if (!newOrder.isEmpty()) oracle.request(account);
// after
_invariant(context, account, newOrder, collateral, protected);
// store
_pendingPosition[context.global.currentId].store(context.currentPosition.global);
_pendingPositions[account][context.local.currentId].store(context.currentPosition.local);
// fund
if (collateral.sign() == 1) token.pull(msg.sender, UFixed18Lib.from(collateral.abs()));
if (collateral.sign() == -1) token.push(msg.sender, UFixed18Lib.from(collateral.abs()));
// events
emit Updated(account, context.currentTimestamp, newMaker, newLong, newShort, collateral, protect);
}
function _loadContext(address account) private view returns (Context memory context) {
// parameters
context.protocolParameter = IMarketFactory(address(factory())).parameter();
context.marketParameter = _parameter.read();
context.riskParameter = _riskParameter.read();
// state
context.global = _global.read();
context.local = _locals[account].read();
// oracle
(context.latestVersion, context.currentTimestamp) = _oracleVersion();
context.positionVersion = _oracleVersionAtPosition(context, _position.read());
}
/// @notice Stores the given context
/// @param context The context to store
/// @param account The account to store for
function _saveContext(Context memory context, address account) private {
_global.store(context.global);
_locals[account].store(context.local);
}
/// @notice Settles the account position up to the latest version
/// @param context The context to use
/// @param account The account to settle
function _settle(Context memory context, address account) private {
context.latestPosition.global = _position.read();
context.latestPosition.local = _positions[account].read();
Position memory nextPosition;
// settle
while (
context.global.currentId != context.global.latestId &&
(nextPosition = _pendingPosition[context.global.latestId + 1].read()).ready(context.latestVersion)
) _processPositionGlobal(context, context.global.latestId + 1, nextPosition);
while (
context.local.currentId != context.local.latestId &&
(nextPosition = _pendingPositions[account][context.local.latestId + 1].read())
.ready(context.latestVersion)
) {
Fixed6 previousDelta = _pendingPositions[account][context.local.latestId].read().delta;
_processPositionLocal(context, account, context.local.latestId + 1, nextPosition);
_checkpointCollateral(context, account, previousDelta, nextPosition);
}
// sync
if (context.latestVersion.timestamp > context.latestPosition.global.timestamp) {
nextPosition = _pendingPosition[context.global.latestId].read();
nextPosition.sync(context.latestVersion);
_processPositionGlobal(context, context.global.latestId, nextPosition);
}
if (context.latestVersion.timestamp > context.latestPosition.local.timestamp) {
nextPosition = _pendingPositions[account][context.local.latestId].read();
nextPosition.sync(context.latestVersion);
_processPositionLocal(context, account, context.local.latestId, nextPosition);
}
_position.store(context.latestPosition.global);
_positions[account].store(context.latestPosition.local);
}
/// @notice Places a collateral checkpoint for the account on the given pending position
/// @param context The context to use
/// @param account The account to checkpoint for
/// @param previousDelta The previous pending position's delta value
/// @param nextPosition The next pending position
function _checkpointCollateral(
Context memory context,
address account,
Fixed6 previousDelta,
Position memory nextPosition
) private {
Position memory latestAccountPosition = _pendingPositions[account][context.local.latestId].read();
Position memory currentAccountPosition = _pendingPositions[account][context.local.currentId].read();
latestAccountPosition.collateral = context.local.collateral
.sub(currentAccountPosition.delta.sub(previousDelta)) // deposits happen after snapshot point
.add(Fixed6Lib.from(nextPosition.fee)) // position fee happens after snapshot point
.add(Fixed6Lib.from(nextPosition.keeper)); // keeper fee happens after snapshot point
_pendingPositions[account][context.local.latestId].store(latestAccountPosition);
}
/// @notice Processes the given global pending position into the latest position
/// @param context The context to use
/// @param newPositionId The id of the pending position to process
/// @param newPosition The pending position to process
function _processPositionGlobal(Context memory context, uint256 newPositionId, Position memory newPosition) private {
Version memory version = _versions[context.latestPosition.global.timestamp].read();
OracleVersion memory oracleVersion = _oracleVersionAtPosition(context, newPosition);
if (!oracleVersion.valid) newPosition.invalidate(context.latestPosition.global);
(uint256 fromTimestamp, uint256 fromId) = (context.latestPosition.global.timestamp, context.global.latestId);
(VersionAccumulationResult memory accumulationResult, UFixed6 accumulatedFee) = version.accumulate(
context.global,
context.latestPosition.global,
newPosition,
context.positionVersion,
oracleVersion,
context.marketParameter,
context.riskParameter
);
context.latestPosition.global.update(newPosition);
context.global.update(newPositionId, oracleVersion.price);
context.global.incrementFees(
accumulatedFee,
newPosition.keeper,
context.marketParameter,
context.protocolParameter
);
context.positionVersion = oracleVersion;
_versions[newPosition.timestamp].store(version);
// events
emit PositionProcessed(
fromTimestamp,
newPosition.timestamp,
fromId,
newPositionId,
accumulationResult
);
}
/// @notice Processes the given local pending position into the latest position
/// @param context The context to use
/// @param account The account to process for
/// @param newPositionId The id of the pending position to process
/// @param newPosition The pending position to process
function _processPositionLocal(
Context memory context,
address account,
uint256 newPositionId,
Position memory newPosition
) private {
Version memory version = _versions[newPosition.timestamp].read();
if (!version.valid) newPosition.invalidate(context.latestPosition.local);
(uint256 fromTimestamp, uint256 fromId) = (context.latestPosition.local.timestamp, context.local.latestId);
LocalAccumulationResult memory accumulationResult = context.local.accumulate(
newPositionId,
context.latestPosition.local,
newPosition,
_versions[context.latestPosition.local.timestamp].read(),
version
);
context.latestPosition.local.update(newPosition);
// events
emit AccountPositionProcessed(
account,
fromTimestamp,
newPosition.timestamp,
fromId,
newPositionId,
accumulationResult
);
}
/// @notice Verifies the invariant of the market
/// @param context The context to use
/// @param account The account to verify the invariant for
/// @param newOrder The order to verify the invariant for
/// @param collateral The collateral change to verify the invariant for
/// @param protected Whether the new position is protected
function _invariant(
Context memory context,
address account,
Order memory newOrder,
Fixed6 collateral,
bool protected
) private view {
// load all pending state
(Position[] memory pendingLocalPositions, Fixed6 collateralAfterFees) = _loadPendingPositions(context, account);
if (protected && (
!context.currentPosition.local.magnitude().isZero() ||
context.latestPosition.local.collateralized(
context.latestVersion,
context.riskParameter,
collateralAfterFees.sub(collateral)
) ||
collateral.lt(Fixed6Lib.from(-1, _liquidationFee(context)))
)) revert MarketInvalidProtectionError();
if (context.currentTimestamp - context.latestVersion.timestamp >= context.riskParameter.staleAfter)
revert MarketStalePriceError();
if (context.marketParameter.closed && newOrder.increasesPosition())
revert MarketClosedError();
if (context.currentPosition.global.maker.gt(context.riskParameter.makerLimit))
revert MarketMakerOverLimitError();
if (!context.currentPosition.local.singleSided())
revert MarketNotSingleSidedError();
if (protected) return; // The following invariants do not apply to protected position updates (liquidations)
if (
msg.sender != account && // sender is operating on own account
!IMarketFactory(address(factory())).operators(account, msg.sender) && // sender is operating on own account
!(newOrder.isEmpty() && collateralAfterFees.isZero() && collateral.gt(Fixed6Lib.ZERO)) // sender is repaying shortfall for this account
) revert MarketOperatorNotAllowedError();
if (
context.global.currentId > context.global.latestId + context.marketParameter.maxPendingGlobal ||
context.local.currentId > context.local.latestId + context.marketParameter.maxPendingLocal
) revert MarketExceedsPendingIdLimitError();
if (
!context.latestPosition.local.collateralized(context.latestVersion, context.riskParameter, collateralAfterFees)
) revert MarketInsufficientCollateralizationError();
for (uint256 i; i < pendingLocalPositions.length; i++)
if (
!pendingLocalPositions[i].collateralized(context.latestVersion, context.riskParameter, collateralAfterFees)
) revert MarketInsufficientCollateralizationError();
if (
(context.local.protection > context.latestPosition.local.timestamp) &&
!newOrder.isEmpty()
) revert MarketProtectedError();
if (
newOrder.liquidityCheckApplicable(context.marketParameter) &&
newOrder.efficiency.lt(Fixed6Lib.ZERO) &&
context.currentPosition.global.efficiency().lt(context.riskParameter.efficiencyLimit)
) revert MarketEfficiencyUnderLimitError();
if (
newOrder.liquidityCheckApplicable(context.marketParameter) &&
context.currentPosition.global.socialized() &&
newOrder.decreasesLiquidity()
) revert MarketInsufficientLiquidityError();
if (collateral.lt(Fixed6Lib.ZERO) && collateralAfterFees.lt(Fixed6Lib.ZERO))
revert MarketInsufficientCollateralError();
}
/// @notice Loads data about all pending positions for the invariant check
/// @param context The context to use
/// @param account The account to load the pending positions for
/// @return pendingLocalPositions All pending positions for the account
/// @return collateralAfterFees The account's collateral after fees
function _loadPendingPositions(
Context memory context,
address account
) private view returns (Position[] memory pendingLocalPositions, Fixed6 collateralAfterFees) {
collateralAfterFees = context.local.collateral;
pendingLocalPositions = new Position[](
context.local.currentId - Math.min(context.local.latestId, context.local.currentId)
);
for (uint256 i; i < pendingLocalPositions.length - 1; i++) {
pendingLocalPositions[i] = _pendingPositions[account][context.local.latestId + 1 + i].read();
collateralAfterFees = collateralAfterFees
.sub(Fixed6Lib.from(pendingLocalPositions[i].fee))
.sub(Fixed6Lib.from(pendingLocalPositions[i].keeper));
}
pendingLocalPositions[pendingLocalPositions.length - 1] = context.currentPosition.local; // current local position hasn't been stored yet
collateralAfterFees = collateralAfterFees
.sub(Fixed6Lib.from(context.currentPosition.local.fee))
.sub(Fixed6Lib.from(context.currentPosition.local.keeper));
}
/// @notice Computes the liquidation fee for the current latest local position
/// @param context The context to use
/// @return The liquidation fee
function _liquidationFee(Context memory context) private view returns (UFixed6) {
return context.latestPosition.local
.liquidationFee(context.latestVersion, context.riskParameter)
.min(UFixed6Lib.from(token.balanceOf()));
}
/// @notice Computes the current oracle status with the market's payoff
/// @return latestVersion The latest oracle version with payoff applied
/// @return currentTimestamp The current oracle timestamp
function _oracleVersion() private view returns (OracleVersion memory latestVersion, uint256 currentTimestamp) {
(latestVersion, currentTimestamp) = oracle.status();
_transform(latestVersion);
}
/// @notice Computes the latest oracle version at a given timestamp with the market's payoff
/// @param timestamp The timestamp to use
/// @return oracleVersion The oracle version at the given timestamp with payoff applied
function _oracleVersionAt(uint256 timestamp) private view returns (OracleVersion memory oracleVersion) {
oracleVersion = oracle.at(timestamp);
_transform(oracleVersion);
}
/// @notice Computes the latest oracle version at a given position with the market's payoff
/// @dev applies the latest valid price when the version at position is invalid
/// @param context The context to use
/// @param toPosition The position to use
/// @return oracleVersion The oracle version at the given position
function _oracleVersionAtPosition(
Context memory context,
Position memory toPosition
) private view returns (OracleVersion memory oracleVersion) {
oracleVersion = _oracleVersionAt(toPosition.timestamp);
if (!oracleVersion.valid) oracleVersion.price = context.global.latestPrice;
}
/// @notice Applies the market's payoff to an oracle version
/// @param oracleVersion The oracle version to transform
function _transform(OracleVersion memory oracleVersion) private view {
if (address(payoff) != address(0)) oracleVersion.price = payoff.payoff(oracleVersion.price);
}
/// @notice Only the coordinator or the owner can call
modifier onlyCoordinator {
if (msg.sender != coordinator && msg.sender != factory().owner()) revert MarketNotCoordinatorError();
_;
}
}