mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #3498 from ethereum/allowthisfselector
Allow `this.f.selector` to be pure.
This commit is contained in:
commit
23484ba6a4
@ -13,6 +13,7 @@ Features:
|
|||||||
variables.
|
variables.
|
||||||
* Syntax Checker: Deprecate the ``var`` keyword (and mark it an error as experimental 0.5.0 feature).
|
* Syntax Checker: Deprecate the ``var`` keyword (and mark it an error as experimental 0.5.0 feature).
|
||||||
* Type Checker: Issue warning for using ``public`` visibility for interface functions.
|
* Type Checker: Issue warning for using ``public`` visibility for interface functions.
|
||||||
|
* Type Checker: Allow `this.f.selector` to be a pure expression.
|
||||||
|
|
||||||
Bugfixes:
|
Bugfixes:
|
||||||
* Error Output: Truncate huge number literals in the middle to avoid output blow-up.
|
* Error Output: Truncate huge number literals in the middle to avoid output blow-up.
|
||||||
|
@ -275,6 +275,22 @@ void ViewPureChecker::endVisit(FunctionCall const& _functionCall)
|
|||||||
reportMutability(mut, _functionCall.location());
|
reportMutability(mut, _functionCall.location());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool ViewPureChecker::visit(MemberAccess const& _memberAccess)
|
||||||
|
{
|
||||||
|
// Catch the special case of `this.f.selector` which is a pure expression.
|
||||||
|
ASTString const& member = _memberAccess.memberName();
|
||||||
|
if (
|
||||||
|
_memberAccess.expression().annotation().type->category() == Type::Category::Function &&
|
||||||
|
member == "selector"
|
||||||
|
)
|
||||||
|
if (auto const* expr = dynamic_cast<MemberAccess const*>(&_memberAccess.expression()))
|
||||||
|
if (auto const* exprInt = dynamic_cast<Identifier const*>(&expr->expression()))
|
||||||
|
if (exprInt->name() == "this")
|
||||||
|
// Do not continue visiting.
|
||||||
|
return false;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void ViewPureChecker::endVisit(MemberAccess const& _memberAccess)
|
void ViewPureChecker::endVisit(MemberAccess const& _memberAccess)
|
||||||
{
|
{
|
||||||
StateMutability mutability = StateMutability::Pure;
|
StateMutability mutability = StateMutability::Pure;
|
||||||
|
@ -56,6 +56,7 @@ private:
|
|||||||
virtual bool visit(ModifierDefinition const& _modifierDef) override;
|
virtual bool visit(ModifierDefinition const& _modifierDef) override;
|
||||||
virtual void endVisit(ModifierDefinition const& _modifierDef) override;
|
virtual void endVisit(ModifierDefinition const& _modifierDef) override;
|
||||||
virtual void endVisit(Identifier const& _identifier) override;
|
virtual void endVisit(Identifier const& _identifier) override;
|
||||||
|
virtual bool visit(MemberAccess const& _memberAccess) override;
|
||||||
virtual void endVisit(MemberAccess const& _memberAccess) override;
|
virtual void endVisit(MemberAccess const& _memberAccess) override;
|
||||||
virtual void endVisit(IndexAccess const& _indexAccess) override;
|
virtual void endVisit(IndexAccess const& _indexAccess) override;
|
||||||
virtual void endVisit(ModifierInvocation const& _modifier) override;
|
virtual void endVisit(ModifierInvocation const& _modifier) override;
|
||||||
|
@ -1014,6 +1014,30 @@ bool ExpressionCompiler::visit(MemberAccess const& _memberAccess)
|
|||||||
_memberAccess.expression().accept(*this);
|
_memberAccess.expression().accept(*this);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
// Another special case for `this.f.selector` which does not need the address.
|
||||||
|
// There are other uses of `.selector` which do need the address, but we want this
|
||||||
|
// specific use to be a pure expression.
|
||||||
|
if (
|
||||||
|
_memberAccess.expression().annotation().type->category() == Type::Category::Function &&
|
||||||
|
member == "selector"
|
||||||
|
)
|
||||||
|
if (auto const* expr = dynamic_cast<MemberAccess const*>(&_memberAccess.expression()))
|
||||||
|
if (auto const* exprInt = dynamic_cast<Identifier const*>(&expr->expression()))
|
||||||
|
if (exprInt->name() == "this")
|
||||||
|
if (Declaration const* declaration = expr->annotation().referencedDeclaration)
|
||||||
|
{
|
||||||
|
u256 identifier;
|
||||||
|
if (auto const* variable = dynamic_cast<VariableDeclaration const*>(declaration))
|
||||||
|
identifier = FunctionType(*variable).externalIdentifier();
|
||||||
|
else if (auto const* function = dynamic_cast<FunctionDefinition const*>(declaration))
|
||||||
|
identifier = FunctionType(*function).externalIdentifier();
|
||||||
|
else
|
||||||
|
solAssert(false, "Contract member is neither variable nor function.");
|
||||||
|
m_context << identifier;
|
||||||
|
/// need to store store it as bytes4
|
||||||
|
utils().leftShiftNumberOnStack(224);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
_memberAccess.expression().accept(*this);
|
_memberAccess.expression().accept(*this);
|
||||||
switch (_memberAccess.expression().annotation().type->category())
|
switch (_memberAccess.expression().annotation().type->category())
|
||||||
|
@ -10219,24 +10219,29 @@ BOOST_AUTO_TEST_CASE(function_types_sig)
|
|||||||
{
|
{
|
||||||
char const* sourceCode = R"(
|
char const* sourceCode = R"(
|
||||||
contract C {
|
contract C {
|
||||||
function f() returns (bytes4) {
|
uint public x;
|
||||||
|
function f() pure returns (bytes4) {
|
||||||
return this.f.selector;
|
return this.f.selector;
|
||||||
}
|
}
|
||||||
function g() returns (bytes4) {
|
function g() returns (bytes4) {
|
||||||
function () external returns (bytes4) fun = this.f;
|
function () pure external returns (bytes4) fun = this.f;
|
||||||
return fun.selector;
|
return fun.selector;
|
||||||
}
|
}
|
||||||
function h() returns (bytes4) {
|
function h() returns (bytes4) {
|
||||||
function () external returns (bytes4) fun = this.f;
|
function () pure external returns (bytes4) fun = this.f;
|
||||||
var funvar = fun;
|
var funvar = fun;
|
||||||
return funvar.selector;
|
return funvar.selector;
|
||||||
}
|
}
|
||||||
|
function i() pure returns (bytes4) {
|
||||||
|
return this.x.selector;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
)";
|
)";
|
||||||
compileAndRun(sourceCode, 0, "C");
|
compileAndRun(sourceCode, 0, "C");
|
||||||
ABI_CHECK(callContractFunction("f()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
ABI_CHECK(callContractFunction("f()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
||||||
ABI_CHECK(callContractFunction("g()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
ABI_CHECK(callContractFunction("g()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
||||||
ABI_CHECK(callContractFunction("h()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
ABI_CHECK(callContractFunction("h()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("f()")).asBytes())));
|
||||||
|
ABI_CHECK(callContractFunction("i()"), encodeArgs(asString(FixedHash<4>(dev::keccak256("x()")).asBytes())));
|
||||||
}
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(constant_string)
|
BOOST_AUTO_TEST_CASE(constant_string)
|
||||||
|
@ -6991,15 +6991,7 @@ BOOST_AUTO_TEST_CASE(function_types_sig)
|
|||||||
CHECK_ERROR(text, TypeError, "Member \"selector\" not found");
|
CHECK_ERROR(text, TypeError, "Member \"selector\" not found");
|
||||||
text = R"(
|
text = R"(
|
||||||
contract C {
|
contract C {
|
||||||
function f() view external returns (bytes4) {
|
function f() pure external returns (bytes4) {
|
||||||
return this.f.selector;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
)";
|
|
||||||
CHECK_SUCCESS_NO_WARNINGS(text);
|
|
||||||
text = R"(
|
|
||||||
contract C {
|
|
||||||
function f() view external returns (bytes4) {
|
|
||||||
return this.f.selector;
|
return this.f.selector;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -326,6 +326,53 @@ BOOST_AUTO_TEST_CASE(function_types)
|
|||||||
CHECK_SUCCESS_NO_WARNINGS(text);
|
CHECK_SUCCESS_NO_WARNINGS(text);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(selector)
|
||||||
|
{
|
||||||
|
string text = R"(
|
||||||
|
contract C {
|
||||||
|
uint public x;
|
||||||
|
function f() payable public {
|
||||||
|
}
|
||||||
|
function g() pure public returns (bytes4) {
|
||||||
|
return this.f.selector ^ this.x.selector;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_SUCCESS_NO_WARNINGS(text);
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(selector_complex)
|
||||||
|
{
|
||||||
|
string text = R"(
|
||||||
|
contract C {
|
||||||
|
function f(C c) pure public returns (C) {
|
||||||
|
return c;
|
||||||
|
}
|
||||||
|
function g() pure public returns (bytes4) {
|
||||||
|
// By passing `this`, we read from the state, even if f itself is pure.
|
||||||
|
return f(this).f.selector;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_ERROR(text, TypeError, "reads from the environment or state and thus requires \"view\"");
|
||||||
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(selector_complex2)
|
||||||
|
{
|
||||||
|
string text = R"(
|
||||||
|
contract C {
|
||||||
|
function f() payable public returns (C) {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
function g() pure public returns (bytes4) {
|
||||||
|
C x = C(0x123);
|
||||||
|
return x.f.selector;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
CHECK_SUCCESS_NO_WARNINGS(text);
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(creation)
|
BOOST_AUTO_TEST_CASE(creation)
|
||||||
{
|
{
|
||||||
string text = R"(
|
string text = R"(
|
||||||
|
Loading…
Reference in New Issue
Block a user