forked from safe-global/safe-smart-account
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Closes safe-global#227: Allow multisend when msg.value > 0; Add call …
…only multisend (safe-global#286)
- Loading branch information
Showing
2 changed files
with
59 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,56 @@ | ||
// SPDX-License-Identifier: LGPL-3.0-only | ||
pragma solidity >=0.7.0 <0.9.0; | ||
|
||
|
||
/// @title Multi Send Call Only - Allows to batch multiple transactions into one, but only calls | ||
/// @author Stefan George - <[email protected]> | ||
/// @author Richard Meissner - <[email protected]> | ||
/// @notice The guard logic is not required here as this contract doesn't support nested delegate calls | ||
contract MultiSendCallOnly { | ||
|
||
/// @dev Sends multiple transactions and reverts all if one fails. | ||
/// @param transactions Encoded transactions. Each transaction is encoded as a packed bytes of | ||
/// operation has to be uint8(0) in this version (=> 1 byte), | ||
/// to as a address (=> 20 bytes), | ||
/// value as a uint256 (=> 32 bytes), | ||
/// data length as a uint256 (=> 32 bytes), | ||
/// data as bytes. | ||
/// see abi.encodePacked for more information on packed encoding | ||
/// @notice The code is for most part the same as the normal MultiSend (to keep compatibility), | ||
/// but reverts if a transaction tries to use a delegatecall. | ||
/// @notice This method is payable as delegatecalls keep the msg.value from the previous call | ||
/// If the calling method (e.g. execTransaction) received ETH this would revert otherwise | ||
function multiSend(bytes memory transactions) | ||
public | ||
payable | ||
{ | ||
// solium-disable-next-line security/no-inline-assembly | ||
assembly { | ||
let length := mload(transactions) | ||
let i := 0x20 | ||
for { } lt(i, length) { } { | ||
// First byte of the data is the operation. | ||
// We shift by 248 bits (256 - 8 [operation byte]) it right since mload will always load 32 bytes (a word). | ||
// This will also zero out unused data. | ||
let operation := shr(0xf8, mload(add(transactions, i))) | ||
// We offset the load address by 1 byte (operation byte) | ||
// We shift it right by 96 bits (256 - 160 [20 address bytes]) to right-align the data and zero out unused data. | ||
let to := shr(0x60, mload(add(transactions, add(i, 0x01)))) | ||
// We offset the load address by 21 byte (operation byte + 20 address bytes) | ||
let value := mload(add(transactions, add(i, 0x15))) | ||
// We offset the load address by 53 byte (operation byte + 20 address bytes + 32 value bytes) | ||
let dataLength := mload(add(transactions, add(i, 0x35))) | ||
// We offset the load address by 85 byte (operation byte + 20 address bytes + 32 value bytes + 32 data length bytes) | ||
let data := add(transactions, add(i, 0x55)) | ||
let success := 0 | ||
switch operation | ||
case 0 { success := call(gas(), to, value, data, dataLength, 0, 0) } | ||
// This version does not allow delegatecalls | ||
case 1 { revert(0, 0) } | ||
if eq(success, 0) { revert(0, 0) } | ||
// Next entry starts at 85 byte + data length | ||
i := add(i, add(0x55, dataLength)) | ||
} | ||
} | ||
} | ||
} |