-
Notifications
You must be signed in to change notification settings - Fork 6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
external contract can waste caller's gas by return value. #12647
Comments
Since the ABI decoder only verifies that returned data is not too short, we could limit the return size to the max size, if the max size is a constant and there are no pointers involved when decoding. In general, I'm not too keen on guarding against hostile called contracts with regards to wasting gas, this is rather complicated and, at least to me, the benefit is not too big. |
it is an attack surface that was used. it will take more effort to protect dynamic-sized return values, but at least avoid exposing staticly-sized return values (like void) |
#12306 would cover this case, though it's a bit more general. And harder to solve. A special case for when there is no data returned at all is much simpler - @chriseth do we want to have such a special case or should it be handled as a part of that more general mechanism for optimizing out unused allocations? |
data:image/s3,"s3://crabby-images/506b4/506b472240c9f910519a91f867c16ce1ad556a8e" alt=":rage4: :rage4:"
I felt the same way as @chriseth when I first saw this (didn't really consider it an "attack") but there's something interesting about this compared to other possible "gas wasting attacks". When one writes Note that this is specifically about As for high level function calls, I think this makes sense as an optimization, but I would not consider it critical (based on the arguments I've seen so far). |
As far as I see it, direct access to gas turned out to be a bad idea, not only because gas costs have changed several times over the past years. Because of that, I would not want to add more options to |
I fully agree. the compiler shouldn't do any gas manipulation. What I do think it should is do LESS UNEEDED, UNEXPECTED work: if a method call returns nothing (or fixed 32 bytes), don't attempt to use/allocate |
When calling a contract, the compiler always assigns a buffer to hold entire returned data, even if not in use.
even when calling a "void" function, the generated yul code always does
regardless if there is a code to use the data or not.
This command, by itself, doesn't cause any significant gas usage. however, any future memory operation will start at higher offset, and the caller contract pays for the entire storage in use, up to the highest offset.
Consider an inner call that does
Not only the inner call takes a lot of gas (as expected, to copy this large buffer), but also the calling code now wastes more gas on its own memory allocations.
Suggestion:
returns (uint)
), use the return buffer size parameter of the call itself, instead ofreturndatasize()
. This way, the caller expects (and uses) an exact amount of memory (it is also slightly cheaper thanreturndatacopy(memptr, returndatasize())
, but this cost saving is marginal)returns ( string memory)
) is more complex. This would probably require a language change, to signal the max expected sizetry/catch blocks
The same rules apply to:
The text was updated successfully, but these errors were encountered: