Skip to content
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

Add builtin erc7201 #15968

Draft
wants to merge 3 commits into
base: develop
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
35 changes: 35 additions & 0 deletions libsolidity/analysis/ConstantEvaluator.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,9 @@
#include <libsolidity/ast/AST.h>
#include <libsolidity/ast/TypeProvider.h>
#include <liblangutil/ErrorReporter.h>
#include <libsolutil/Keccak256.h>
#include <libsolutil/StringUtils.h>
#include <libsolutil/FixedHash.h>

#include <limits>

Expand Down Expand Up @@ -407,3 +410,35 @@ void ConstantEvaluator::endVisit(TupleExpression const& _tuple)
if (!_tuple.isInlineArray() && _tuple.components().size() == 1)
m_values[&_tuple] = evaluate(*_tuple.components().front());
}

void ConstantEvaluator::endVisit(FunctionCall const& _functionCall)
{
using util::keccak256;
using util::h256;

auto const* builtinFunction = dynamic_cast<MagicVariableDeclaration const*>(ASTNode::referencedDeclaration(_functionCall.expression()));
if (!builtinFunction)
return;

auto const* functionType = builtinFunction->functionType(true);
solAssert(functionType);
switch (functionType->kind())
{
case FunctionType::Kind::ERC7201:
{
solAssert(_functionCall.arguments().size() == 1);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Shouldn't we report an error instead?

Copy link
Collaborator Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That is already checked in TypeChecker::typeCheckFunctionGeneralChecks.
I think also it doesn't make sense to raise such error here.
But your question touches on something I am uncomfortable still in extending the ConstantEvaluator, which is depending/assuming that analysis step already passed.

auto const* argument = dynamic_cast<Literal const*>(_functionCall.arguments()[0].get());
solAssert(argument);
ASTString argStringLiteral = argument->valueWithoutUnderscores();
auto value =
u256(keccak256(h256(u256(keccak256(argStringLiteral)) - 1))) &
Copy link
Member

@r0qs r0qs Mar 26, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why the u256 conversion here? Isn't better to work with bytes and only convert everything to u256 when necessary?

Copy link
Collaborator Author

@matheusaaguiar matheusaaguiar Mar 26, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I have to check that but looks like probably those conversions can be skipped.
I just tentatively reconstructed from the solidity spec here.
At least the computed values from both versions (cpp and solidity) were matching in the test I did in remix.

u256(~0xff)
;
solAssert(functionType->returnParameterTypes().size() == 1);
m_values[&_functionCall] = TypedRational{functionType->returnParameterTypes()[0], rational{value}};
break;
}
default:
break;
}
}
1 change: 1 addition & 0 deletions libsolidity/analysis/ConstantEvaluator.h
Original file line number Diff line number Diff line change
Expand Up @@ -79,6 +79,7 @@ class ConstantEvaluator: private ASTConstVisitor
void endVisit(Literal const& _literal) override;
void endVisit(Identifier const& _identifier) override;
void endVisit(TupleExpression const& _tuple) override;
void endVisit(FunctionCall const& _functionCall) override;

langutil::ErrorReporter& m_errorReporter;
/// Current recursion depth.
Expand Down
4 changes: 3 additions & 1 deletion libsolidity/analysis/GlobalContext.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -62,7 +62,8 @@ int magicVariableToID(std::string const& _name)
{"tx", -26},
{"type", -27},
{"this", -28},
{"blobhash", -29}
{"blobhash", -29},
{"erc7201", -30}
};

if (auto id = magicVariables.find(_name); id != magicVariables.end())
Expand All @@ -83,6 +84,7 @@ inline std::vector<std::shared_ptr<MagicVariableDeclaration const>> constructMag
magicVarDecl("block", TypeProvider::magic(MagicType::Kind::Block)),
magicVarDecl("blockhash", TypeProvider::function(strings{"uint256"}, strings{"bytes32"}, FunctionType::Kind::BlockHash, StateMutability::View)),
magicVarDecl("ecrecover", TypeProvider::function(strings{"bytes32", "uint8", "bytes32", "bytes32"}, strings{"address"}, FunctionType::Kind::ECRecover, StateMutability::Pure)),
magicVarDecl("erc7201", TypeProvider::function(strings{"bytes memory"}, strings{"uint256"}, FunctionType::Kind::ERC7201, StateMutability::Pure)),
magicVarDecl("gasleft", TypeProvider::function(strings(), strings{"uint256"}, FunctionType::Kind::GasLeft, StateMutability::View)),
magicVarDecl("keccak256", TypeProvider::function(strings{"bytes memory"}, strings{"bytes32"}, FunctionType::Kind::KECCAK256, StateMutability::Pure)),
magicVarDecl("msg", TypeProvider::magic(MagicType::Kind::Message)),
Expand Down
1 change: 0 additions & 1 deletion libsolidity/analysis/PostTypeContractLevelChecker.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -91,7 +91,6 @@ void PostTypeContractLevelChecker::checkStorageLayoutSpecifier(ContractDefinitio

if (!*baseSlotExpression.annotation().isPure)
{
// TODO: introduce and handle erc7201 as a builtin function
m_errorReporter.typeError(
1139_error,
baseSlotExpression.location(),
Expand Down
4 changes: 3 additions & 1 deletion libsolidity/ast/Types.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -3125,6 +3125,7 @@ std::string FunctionType::richIdentifier() const
case Kind::ABIDecode: id += "abidecode"; break;
case Kind::BlobHash: id += "blobhash"; break;
case Kind::MetaType: id += "metatype"; break;
case Kind::ERC7201: id += "erc7201"; break;
}
id += "_" + stateMutabilityToString(m_stateMutability);
id += identifierList(m_parameterTypes) + "returns" + identifierList(m_returnParameterTypes);
Expand Down Expand Up @@ -3713,7 +3714,8 @@ bool FunctionType::isPure() const
m_kind == Kind::ABIDecode ||
m_kind == Kind::MetaType ||
m_kind == Kind::Wrap ||
m_kind == Kind::Unwrap;
m_kind == Kind::Unwrap ||
m_kind == Kind::ERC7201;
}

TypePointers FunctionType::parseElementaryTypeVector(strings const& _types)
Expand Down
1 change: 1 addition & 0 deletions libsolidity/ast/Types.h
Original file line number Diff line number Diff line change
Expand Up @@ -1297,6 +1297,7 @@ class FunctionType: public Type
ABIDecode,
GasLeft, ///< gasleft()
MetaType, ///< type(...)
ERC7201, ///< erc7201(...)
/// Refers to a function declaration without calling context
/// (i.e. when accessed directly via the name of the containing contract).
/// Cannot be called.
Expand Down
3 changes: 3 additions & 0 deletions libsolidity/codegen/ExpressionCompiler.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -1521,6 +1521,9 @@ bool ExpressionCompiler::visit(FunctionCall const& _functionCall)
case FunctionType::Kind::MetaType:
// No code to generate.
break;
case FunctionType::Kind::ERC7201:
solUnimplementedAssert(false, "Codegen does not support erc7201 builtin yet.");
break;
}
}
return false;
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
contract C {
uint constant x = erc7201("A");
uint[x] array;
}
// ----
// Warning 7325: (53-60): Type uint256[36579005187129934694193755934841191771209741707776365283473783080460440925696] covers a large part of storage and thus makes collisions likely. Either use mappings or dynamic arrays and allow their size to be increased only in small quantities per transaction.
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
contract C {
function f() public pure returns (uint) {
return erc7201("ABC");
}
}
// ----
// UnimplementedFeatureError 1834: (0-97): Codegen does not support erc7201 builtin yet.
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
function erc7201() pure returns (uint) {
return 42;
}
// ----
// Warning 2319: (0-57): This declaration shadows a builtin symbol.
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
contract C {
function f() public pure {
erc7201;
}
}
// ----
// Warning 6133: (52-59): Statement has no effect.
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
contract C {
uint erc7201;
}
// ----
// Warning 2319: (17-29): This declaration shadows a builtin symbol.
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
contract C {
uint x = erc7201();
uint y = erc7201("12", "34");
uint z = erc7201("A", "BC", "D");
}
// ----
// TypeError 6160: (26-35): Wrong argument count for function call: 0 arguments given but expected 1.
// TypeError 6160: (50-69): Wrong argument count for function call: 2 arguments given but expected 1.
// TypeError 6160: (84-107): Wrong argument count for function call: 3 arguments given but expected 1.
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
contract C {
uint constant x = erc7201("abc");
function f() public pure returns(uint) {
return x;
}
}
// ----
// UnimplementedFeatureError 1834: (0-121): Codegen does not support erc7201 builtin yet.
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
contract C {
function f() public {
erc7201.gas();
}
}
// ----
// TypeError 9582: (47-58): Member "gas" not found or not visible after argument-dependent lookup in function (bytes memory) pure returns (uint256).
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
contract C {
function f() public {
erc7201.value();
}
}
// ----
// TypeError 8820: (47-60): Member "value" is only available for payable functions.