@@ -89,6 +89,45 @@ contract GatewayZEVM is
89
89
_unpause ();
90
90
}
91
91
92
+ /// @notice Helper function to safely execute transferFrom
93
+ /// @param zrc20 The ZRC20 token address
94
+ /// @param from The sender address
95
+ /// @param to The recipient address
96
+ /// @param amount The amount to transfer
97
+ /// @return True if the transfer was successful, false otherwise.
98
+ function _safeTransferFrom (address zrc20 , address from , address to , uint256 amount ) private returns (bool ) {
99
+ try IZRC20 (zrc20).transferFrom (from, to, amount) returns (bool success ) {
100
+ return success;
101
+ } catch {
102
+ return false ;
103
+ }
104
+ }
105
+
106
+ // @notice Helper function to safely burn ZRC20 tokens
107
+ // @param zrc20 The ZRC20 token address
108
+ // @param amount The amount to burn
109
+ // @return True if the burn was successful, false otherwise
110
+ function _safeBurn (address zrc20 , uint256 amount ) private returns (bool ) {
111
+ try IZRC20 (zrc20).burn (amount) returns (bool success ) {
112
+ return success;
113
+ } catch {
114
+ return false ;
115
+ }
116
+ }
117
+
118
+ // @notice Helper function to safely deposit
119
+ // @param zrc20 The ZRC20 token address
120
+ // @param amount The target address to receive the deposited tokens
121
+ // @param amount The amount to deposit
122
+ // @return True if the deposit was successful, false otherwise
123
+ function _safeDeposit (address zrc20 , address target , uint256 amount ) private returns (bool ) {
124
+ try IZRC20 (zrc20).deposit (target, amount) returns (bool success ) {
125
+ return success;
126
+ } catch {
127
+ return false ;
128
+ }
129
+ }
130
+
92
131
/// @dev Private function to withdraw ZRC20 tokens.
93
132
/// @param amount The amount of tokens to withdraw.
94
133
/// @param zrc20 The address of the ZRC20 token.
@@ -105,15 +144,18 @@ contract GatewayZEVM is
105
144
/// @return The gas fee for the withdrawal.
106
145
function _withdrawZRC20WithGasLimit (uint256 amount , address zrc20 , uint256 gasLimit ) private returns (uint256 ) {
107
146
(address gasZRC20 , uint256 gasFee ) = IZRC20 (zrc20).withdrawGasFeeWithGasLimit (gasLimit);
108
- if (! IZRC20 (gasZRC20).transferFrom (msg .sender , PROTOCOL_ADDRESS, gasFee)) {
109
- revert GasFeeTransferFailed ();
147
+
148
+ if (! _safeTransferFrom (gasZRC20, msg .sender , PROTOCOL_ADDRESS, gasFee)) {
149
+ revert GasFeeTransferFailed (gasZRC20, PROTOCOL_ADDRESS, gasFee);
110
150
}
111
151
112
- if (! IZRC20 (zrc20). transferFrom ( msg .sender , address (this ), amount)) {
113
- revert ZRC20TransferFailed ();
152
+ if (! _safeTransferFrom (zrc20, msg .sender , address (this ), amount)) {
153
+ revert ZRC20TransferFailed (zrc20, msg . sender , address ( this ), amount );
114
154
}
115
155
116
- if (! IZRC20 (zrc20).burn (amount)) revert ZRC20BurnFailed ();
156
+ if (! _safeBurn (zrc20, amount)) {
157
+ revert ZRC20BurnFailed (zrc20, amount);
158
+ }
117
159
118
160
return gasFee;
119
161
}
@@ -122,10 +164,15 @@ contract GatewayZEVM is
122
164
/// @param amount The amount of tokens to transfer.
123
165
/// @param to The address to transfer the tokens to.
124
166
function _transferZETA (uint256 amount , address to ) private {
125
- if (! IWETH9 (zetaToken).transferFrom (msg .sender , address (this ), amount)) revert FailedZetaSent ();
126
- IWETH9 (zetaToken).withdraw (amount);
127
- (bool sent ,) = to.call { value: amount }("" );
128
- if (! sent) revert FailedZetaSent ();
167
+ if (! _safeTransferFrom (zetaToken, msg .sender , address (this ), amount)) {
168
+ revert FailedZetaSent (address (this ), amount);
169
+ }
170
+ try IWETH9 (zetaToken).withdraw (amount) {
171
+ (bool sent ,) = to.call { value: amount }("" );
172
+ if (! sent) revert FailedZetaSent (to, amount);
173
+ } catch {
174
+ revert FailedZetaSent (to, amount);
175
+ }
129
176
}
130
177
131
178
/// @notice Withdraw ZRC20 tokens to an external chain.
@@ -144,7 +191,9 @@ contract GatewayZEVM is
144
191
{
145
192
if (receiver.length == 0 ) revert ZeroAddress ();
146
193
if (amount == 0 ) revert InsufficientZRC20Amount ();
147
- if (revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded ();
194
+ if (revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
195
+ revert MessageSizeExceeded (revertOptions.revertMessage.length , MAX_MESSAGE_SIZE);
196
+ }
148
197
149
198
uint256 gasFee = _withdrawZRC20 (amount, zrc20);
150
199
emit Withdrawn (
@@ -182,7 +231,9 @@ contract GatewayZEVM is
182
231
if (receiver.length == 0 ) revert ZeroAddress ();
183
232
if (amount == 0 ) revert InsufficientZRC20Amount ();
184
233
if (callOptions.gasLimit == 0 ) revert InsufficientGasLimit ();
185
- if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded ();
234
+ if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
235
+ revert MessageSizeExceeded (message.length + revertOptions.revertMessage.length , MAX_MESSAGE_SIZE);
236
+ }
186
237
187
238
uint256 gasFee = _withdrawZRC20WithGasLimit (amount, zrc20, callOptions.gasLimit);
188
239
emit WithdrawnAndCalled (
@@ -289,7 +340,9 @@ contract GatewayZEVM is
289
340
whenNotPaused
290
341
{
291
342
if (callOptions.gasLimit == 0 ) revert InsufficientGasLimit ();
292
- if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) revert MessageSizeExceeded ();
343
+ if (message.length + revertOptions.revertMessage.length > MAX_MESSAGE_SIZE) {
344
+ revert MessageSizeExceeded (message.length + revertOptions.revertMessage.length , MAX_MESSAGE_SIZE);
345
+ }
293
346
294
347
_call (receiver, zrc20, message, callOptions, revertOptions);
295
348
}
@@ -306,8 +359,8 @@ contract GatewayZEVM is
306
359
if (receiver.length == 0 ) revert ZeroAddress ();
307
360
308
361
(address gasZRC20 , uint256 gasFee ) = IZRC20 (zrc20).withdrawGasFeeWithGasLimit (callOptions.gasLimit);
309
- if (! IZRC20 (gasZRC20). transferFrom ( msg .sender , PROTOCOL_ADDRESS, gasFee)) {
310
- revert GasFeeTransferFailed ();
362
+ if (! _safeTransferFrom (gasZRC20, msg .sender , PROTOCOL_ADDRESS, gasFee)) {
363
+ revert GasFeeTransferFailed (gasZRC20, PROTOCOL_ADDRESS, gasFee );
311
364
}
312
365
313
366
emit Called (msg .sender , zrc20, receiver, message, callOptions, revertOptions);
@@ -323,7 +376,9 @@ contract GatewayZEVM is
323
376
324
377
if (target == PROTOCOL_ADDRESS || target == address (this )) revert InvalidTarget ();
325
378
326
- if (! IZRC20 (zrc20).deposit (target, amount)) revert ZRC20DepositFailed ();
379
+ if (! _safeDeposit (zrc20, target, amount)) {
380
+ revert ZRC20DepositFailed (zrc20, target, amount);
381
+ }
327
382
}
328
383
329
384
/// @notice Execute a user-specified contract on ZEVM.
@@ -371,7 +426,10 @@ contract GatewayZEVM is
371
426
if (amount == 0 ) revert InsufficientZRC20Amount ();
372
427
if (target == PROTOCOL_ADDRESS || target == address (this )) revert InvalidTarget ();
373
428
374
- if (! IZRC20 (zrc20).deposit (target, amount)) revert ZRC20DepositFailed ();
429
+ if (! _safeDeposit (zrc20, target, amount)) {
430
+ revert ZRC20DepositFailed (zrc20, target, amount);
431
+ }
432
+
375
433
UniversalContract (target).onCall (context, zrc20, amount, message);
376
434
}
377
435
@@ -436,7 +494,10 @@ contract GatewayZEVM is
436
494
if (amount == 0 ) revert InsufficientZRC20Amount ();
437
495
if (target == PROTOCOL_ADDRESS || target == address (this )) revert InvalidTarget ();
438
496
439
- if (! IZRC20 (zrc20).deposit (target, amount)) revert ZRC20DepositFailed ();
497
+ if (! _safeDeposit (zrc20, target, amount)) {
498
+ revert ZRC20DepositFailed (zrc20, target, amount);
499
+ }
500
+
440
501
Revertable (target).onRevert (revertContext);
441
502
}
442
503
0 commit comments