Documentation for function call options.

This commit is contained in:
chriseth 2020-01-23 13:17:46 +01:00
parent 893fb4d05b
commit 981ed5f773
4 changed files with 96 additions and 19 deletions

View File

@ -335,7 +335,7 @@ operations as long as there is enough gas passed on to it.
:: ::
pragma solidity ^0.6.0; pragma solidity >0.6.1 <0.7.0;
contract Test { contract Test {
// This function is called for all messages sent to // This function is called for all messages sent to
@ -382,7 +382,7 @@ operations as long as there is enough gas passed on to it.
(bool success,) = address(test).call(abi.encodeWithSignature("nonExistingFunction()")); (bool success,) = address(test).call(abi.encodeWithSignature("nonExistingFunction()"));
require(success); require(success);
// results in test.x becoming == 1 and test.y becoming 0. // results in test.x becoming == 1 and test.y becoming 0.
(success,) = address(test).call.value(1)(abi.encodeWithSignature("nonExistingFunction()")); (success,) = address(test).call{value: 1}(abi.encodeWithSignature("nonExistingFunction()"));
require(success); require(success);
// results in test.x becoming == 1 and test.y becoming 1. // results in test.x becoming == 1 and test.y becoming 1.

View File

@ -75,9 +75,10 @@ all function arguments have to be copied to memory.
it is a message call as part of the overall transaction. it is a message call as part of the overall transaction.
When calling functions of other contracts, you can specify the amount of Wei or When calling functions of other contracts, you can specify the amount of Wei or
gas sent with the call with the special options ``.value()`` and ``.gas()``, gas sent with the call with the special options ``{value: 10, gas: 10000}``.
respectively. Any Wei you send to the contract is added to the total balance Note that it is discouraged to specify gas values explicitly, since the gas costs
of the contract: of opcodes can change in the future. Any Wei you send to the contract is added
to the total balance of that contract:
:: ::
@ -90,14 +91,14 @@ of the contract:
contract Consumer { contract Consumer {
InfoFeed feed; InfoFeed feed;
function setFeed(InfoFeed addr) public { feed = addr; } function setFeed(InfoFeed addr) public { feed = addr; }
function callFeed() public { feed.info.value(10).gas(800)(); } function callFeed() public { feed.info{value: 10, gas: 800}(); }
} }
You need to use the modifier ``payable`` with the ``info`` function because You need to use the modifier ``payable`` with the ``info`` function because
otherwise, the ``.value()`` option would not be available. otherwise, the ``value`` option would not be available.
.. warning:: .. warning::
Be careful that ``feed.info.value(10).gas(800)`` only locally sets the Be careful that ``feed.info{value: 10, gas: 800}`` only locally sets the
``value`` and amount of ``gas`` sent with the function call, and the ``value`` and amount of ``gas`` sent with the function call, and the
parentheses at the end perform the actual call. So in this case, the parentheses at the end perform the actual call. So in this case, the
function is not called and the ``value`` and ``gas`` settings are lost. function is not called and the ``value`` and ``gas`` settings are lost.
@ -121,6 +122,11 @@ throws an exception or goes out of gas.
external functions happen after any changes to state variables in your contract external functions happen after any changes to state variables in your contract
so your contract is not vulnerable to a reentrancy exploit. so your contract is not vulnerable to a reentrancy exploit.
.. note::
Before Solidity 0.6.2, the recommended way to specify the value and gas
was to use ``f.value(x).gas(g)()``. This is still possible but deprecated
and will be removed with Solidity 0.7.0.
Named Calls and Anonymous Function Parameters Named Calls and Anonymous Function Parameters
--------------------------------------------- ---------------------------------------------
@ -196,17 +202,81 @@ is compiled so recursive creation-dependencies are not possible.
function createAndEndowD(uint arg, uint amount) public payable { function createAndEndowD(uint arg, uint amount) public payable {
// Send ether along with the creation // Send ether along with the creation
D newD = (new D).value(amount)(arg); D newD = new D{value: amount}(arg);
newD.x(); newD.x();
} }
} }
As seen in the example, it is possible to send Ether while creating As seen in the example, it is possible to send Ether while creating
an instance of ``D`` using the ``.value()`` option, but it is not possible an instance of ``D`` using the ``value`` option, but it is not possible
to limit the amount of gas. to limit the amount of gas.
If the creation fails (due to out-of-stack, not enough balance or other problems), If the creation fails (due to out-of-stack, not enough balance or other problems),
an exception is thrown. an exception is thrown.
Salted contract creations / create2
-----------------------------------
When creating a contract, the address of the contract is computed from
the address of the creating contract and a counter that is increased with
each contract creation.
If you specify the option ``salt`` (a bytes32 value), then contract creation will
use a different mechanism to come up with the address of the new contract:
It will compute the address from the address of the creating contract,
the given salt value, the (creation) bytecode of the created contract and the constructor
arguments.
In particular, the counter ("nonce") is not used. This allows for more flexibility
in creating contracts: You are able to derive the address of the
new contract before it is created. Furthermore, you can rely on this address
also in case the creating
contracts creates other contracts in the meantime.
The main use-case here is contracts that act as judges for off-chain interactions,
which only need to be created if there is a dispute.
::
pragma solidity >0.6.1 <0.7.0;
contract D {
uint public x;
constructor(uint a) public {
x = a;
}
}
contract C {
function createDSalted(bytes32 salt, uint arg) public {
/// This complicated expression just tells you how the address
/// can be pre-computed. It is just there for illustration.
/// You actually only need ``new D{salt: salt}(arg)``.
address predictedAddress = address(bytes20(keccak256(abi.encodePacked(
byte(0xff),
address(this),
salt,
keccak256(abi.encodePacked(
type(D).creationCode,
arg
))
))));
D d = new D{salt: salt}(arg);
require(address(d) == predictedAddress);
}
}
.. warning::
There are some peculiarities in relation to salted creation. A contract can be
re-created at the same address after having been destroyed. Yet, it is possible
for that newly created contract to have a different deployed bytecode even
though the creation bytecode has been the same (which is a requirement because
otherwise the address would change). This is due to the fact that the compiler
can query external state that might have changed between the two creations
and incorporate that into the deployed bytecode before it is stored.
Order of Evaluation of Expressions Order of Evaluation of Expressions
================================== ==================================

View File

@ -89,7 +89,7 @@ as it uses ``call`` which forwards all remaining gas by default:
mapping(address => uint) shares; mapping(address => uint) shares;
/// Withdraw your share. /// Withdraw your share.
function withdraw() public { function withdraw() public {
(bool success,) = msg.sender.call.value(shares[msg.sender])(""); (bool success,) = msg.sender.call{value: shares[msg.sender]}("");
if (success) if (success)
shares[msg.sender] = 0; shares[msg.sender] = 0;
} }
@ -149,7 +149,7 @@ Sending and Receiving Ether
(for example in the "details" section in Remix). (for example in the "details" section in Remix).
- There is a way to forward more gas to the receiving contract using - There is a way to forward more gas to the receiving contract using
``addr.call.value(x)("")``. This is essentially the same as ``addr.transfer(x)``, ``addr.call{value: x}("")``. This is essentially the same as ``addr.transfer(x)``,
only that it forwards all remaining gas and opens up the ability for the only that it forwards all remaining gas and opens up the ability for the
recipient to perform more expensive actions (and it returns a failure code recipient to perform more expensive actions (and it returns a failure code
instead of automatically propagating the error). This might include calling back instead of automatically propagating the error). This might include calling back

View File

@ -276,17 +276,17 @@ Example::
arbitrary arguments and would also handle a first argument of type arbitrary arguments and would also handle a first argument of type
``bytes4`` differently. These edge cases were removed in version 0.5.0. ``bytes4`` differently. These edge cases were removed in version 0.5.0.
It is possible to adjust the supplied gas with the ``.gas()`` modifier:: It is possible to adjust the supplied gas with the ``gas`` modifier::
address(nameReg).call.gas(1000000)(abi.encodeWithSignature("register(string)", "MyName")); address(nameReg).call{gas: 1000000}(abi.encodeWithSignature("register(string)", "MyName"));
Similarly, the supplied Ether value can be controlled too:: Similarly, the supplied Ether value can be controlled too::
address(nameReg).call.value(1 ether)(abi.encodeWithSignature("register(string)", "MyName")); address(nameReg).call{value: 1 ether}(abi.encodeWithSignature("register(string)", "MyName"));
Lastly, these modifiers can be combined. Their order does not matter:: Lastly, these modifiers can be combined. Their order does not matter::
address(nameReg).call.gas(1000000).value(1 ether)(abi.encodeWithSignature("register(string)", "MyName")); address(nameReg).call{gas: 1000000, value: 1 ether}(abi.encodeWithSignature("register(string)", "MyName"));
In a similar way, the function ``delegatecall`` can be used: the difference is that only the code of the given address is used, all other aspects (storage, balance, ...) are taken from the current contract. The purpose of ``delegatecall`` is to use library code which is stored in another contract. The user has to ensure that the layout of storage in both contracts is suitable for delegatecall to be used. In a similar way, the function ``delegatecall`` can be used: the difference is that only the code of the given address is used, all other aspects (storage, balance, ...) are taken from the current contract. The purpose of ``delegatecall`` is to use library code which is stored in another contract. The user has to ensure that the layout of storage in both contracts is suitable for delegatecall to be used.
@ -297,7 +297,8 @@ Since byzantium ``staticcall`` can be used as well. This is basically the same a
All three functions ``call``, ``delegatecall`` and ``staticcall`` are very low-level functions and should only be used as a *last resort* as they break the type-safety of Solidity. All three functions ``call``, ``delegatecall`` and ``staticcall`` are very low-level functions and should only be used as a *last resort* as they break the type-safety of Solidity.
The ``.gas()`` option is available on all three methods, while the ``.value()`` option is not supported for ``delegatecall``. The ``gas`` option is available on all three methods, while the ``value`` option is not
supported for ``delegatecall``.
.. note:: .. note::
All contracts can be converted to ``address`` type, so it is possible to query the balance of the All contracts can be converted to ``address`` type, so it is possible to query the balance of the
@ -635,8 +636,12 @@ External (or public) functions have the following members:
* ``.address`` returns the address of the contract of the function. * ``.address`` returns the address of the contract of the function.
* ``.selector`` returns the :ref:`ABI function selector <abi_function_selector>` * ``.selector`` returns the :ref:`ABI function selector <abi_function_selector>`
* ``.gas(uint)`` returns a callable function object which, when called, will send the specified amount of gas to the target function. See :ref:`External Function Calls <external-function-calls>` for more information. * ``.gas(uint)`` returns a callable function object which, when called, will send
* ``.value(uint)`` returns a callable function object which, when called, will send the specified amount of wei to the target function. See :ref:`External Function Calls <external-function-calls>` for more information. the specified amount of gas to the target function. Deprecated - use ``{gas: ...}`` instead.
See :ref:`External Function Calls <external-function-calls>` for more information.
* ``.value(uint)`` returns a callable function object which, when called, will
send the specified amount of wei to the target function. Deprecated - use ``{value: ...}`` instead.
See :ref:`External Function Calls <external-function-calls>` for more information.
Example that shows how to use the members:: Example that shows how to use the members::
@ -651,6 +656,8 @@ Example that shows how to use the members::
function g() public { function g() public {
this.f.gas(10).value(800)(); this.f.gas(10).value(800)();
// New syntax:
// this.f{gas: 10, value: 800}()
} }
} }