mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Support referencing other constants in inline assembly
This commit is contained in:
parent
2a1b6f55af
commit
4d90180e24
@ -3,6 +3,7 @@
|
|||||||
Language Features:
|
Language Features:
|
||||||
* Allow to obtain the selector of public or external library functions via a member ``.selector``.
|
* Allow to obtain the selector of public or external library functions via a member ``.selector``.
|
||||||
* Parser: Allow splitting string and hexadecimal string literals into multiple parts.
|
* Parser: Allow splitting string and hexadecimal string literals into multiple parts.
|
||||||
|
* Inline Assembly: Support referencing other constants.
|
||||||
|
|
||||||
|
|
||||||
Compiler Features:
|
Compiler Features:
|
||||||
|
@ -41,6 +41,8 @@ set(sources
|
|||||||
ast/ASTJsonConverter.h
|
ast/ASTJsonConverter.h
|
||||||
ast/ASTPrinter.cpp
|
ast/ASTPrinter.cpp
|
||||||
ast/ASTPrinter.h
|
ast/ASTPrinter.h
|
||||||
|
ast/ASTUtils.cpp
|
||||||
|
ast/ASTUtils.h
|
||||||
ast/ASTVisitor.h
|
ast/ASTVisitor.h
|
||||||
ast/ExperimentalFeatures.h
|
ast/ExperimentalFeatures.h
|
||||||
ast/Types.cpp
|
ast/Types.cpp
|
||||||
|
@ -22,6 +22,7 @@
|
|||||||
|
|
||||||
#include <libsolidity/analysis/TypeChecker.h>
|
#include <libsolidity/analysis/TypeChecker.h>
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
|
#include <libsolidity/ast/ASTUtils.h>
|
||||||
#include <libsolidity/ast/TypeProvider.h>
|
#include <libsolidity/ast/TypeProvider.h>
|
||||||
|
|
||||||
#include <libyul/AsmAnalysis.h>
|
#include <libyul/AsmAnalysis.h>
|
||||||
@ -641,6 +642,8 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
solAssert(var->type(), "Expected variable type!");
|
solAssert(var->type(), "Expected variable type!");
|
||||||
if (var->isConstant())
|
if (var->isConstant())
|
||||||
{
|
{
|
||||||
|
var = rootVariableDeclaration(*var);
|
||||||
|
|
||||||
if (!var->value())
|
if (!var->value())
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(_identifier.location, "Constant has no value.");
|
m_errorReporter.typeError(_identifier.location, "Constant has no value.");
|
||||||
@ -651,7 +654,7 @@ bool TypeChecker::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
type(*var->value())->category() != Type::Category::RationalNumber
|
type(*var->value())->category() != Type::Category::RationalNumber
|
||||||
))
|
))
|
||||||
{
|
{
|
||||||
m_errorReporter.typeError(_identifier.location, "Only direct number constants are supported by inline assembly.");
|
m_errorReporter.typeError(_identifier.location, "Only direct number constants and references to such constants are supported by inline assembly.");
|
||||||
return size_t(-1);
|
return size_t(-1);
|
||||||
}
|
}
|
||||||
else if (_context == yul::IdentifierContext::LValue)
|
else if (_context == yul::IdentifierContext::LValue)
|
||||||
|
42
libsolidity/ast/ASTUtils.cpp
Normal file
42
libsolidity/ast/ASTUtils.cpp
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
/*
|
||||||
|
This file is part of solidity.
|
||||||
|
|
||||||
|
solidity is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
solidity is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <libsolidity/ast/AST.h>
|
||||||
|
#include <libsolidity/ast/ASTUtils.h>
|
||||||
|
|
||||||
|
namespace dev
|
||||||
|
{
|
||||||
|
namespace solidity
|
||||||
|
{
|
||||||
|
|
||||||
|
VariableDeclaration const* rootVariableDeclaration(VariableDeclaration const& _varDecl)
|
||||||
|
{
|
||||||
|
solAssert(_varDecl.isConstant(), "Constant variable expected");
|
||||||
|
|
||||||
|
VariableDeclaration const* rootDecl = &_varDecl;
|
||||||
|
Identifier const* identifier;
|
||||||
|
while ((identifier = dynamic_cast<Identifier const*>(rootDecl->value().get())))
|
||||||
|
{
|
||||||
|
auto referencedVarDecl = dynamic_cast<VariableDeclaration const*>(identifier->annotation().referencedDeclaration);
|
||||||
|
solAssert(referencedVarDecl && referencedVarDecl->isConstant(), "Identifier is not referencing a variable declaration");
|
||||||
|
rootDecl = referencedVarDecl;
|
||||||
|
}
|
||||||
|
return rootDecl;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
30
libsolidity/ast/ASTUtils.h
Normal file
30
libsolidity/ast/ASTUtils.h
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
/*
|
||||||
|
This file is part of solidity.
|
||||||
|
|
||||||
|
solidity is free software: you can redistribute it and/or modify
|
||||||
|
it under the terms of the GNU General Public License as published by
|
||||||
|
the Free Software Foundation, either version 3 of the License, or
|
||||||
|
(at your option) any later version.
|
||||||
|
|
||||||
|
solidity is distributed in the hope that it will be useful,
|
||||||
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
GNU General Public License for more details.
|
||||||
|
|
||||||
|
You should have received a copy of the GNU General Public License
|
||||||
|
along with solidity. If not, see <http://www.gnu.org/licenses/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
namespace dev
|
||||||
|
{
|
||||||
|
namespace solidity
|
||||||
|
{
|
||||||
|
|
||||||
|
/// Find the topmost referenced variable declaration when the given variable
|
||||||
|
/// declaration value is an identifier. Works only for constant variable declarations.
|
||||||
|
VariableDeclaration const* rootVariableDeclaration(VariableDeclaration const& _varDecl);
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -21,6 +21,7 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include <libsolidity/ast/AST.h>
|
#include <libsolidity/ast/AST.h>
|
||||||
|
#include <libsolidity/ast/ASTUtils.h>
|
||||||
#include <libsolidity/ast/TypeProvider.h>
|
#include <libsolidity/ast/TypeProvider.h>
|
||||||
#include <libsolidity/codegen/CompilerUtils.h>
|
#include <libsolidity/codegen/CompilerUtils.h>
|
||||||
#include <libsolidity/codegen/ContractCompiler.h>
|
#include <libsolidity/codegen/ContractCompiler.h>
|
||||||
@ -631,6 +632,7 @@ bool ContractCompiler::visit(InlineAssembly const& _inlineAssembly)
|
|||||||
{
|
{
|
||||||
if (variable->isConstant())
|
if (variable->isConstant())
|
||||||
{
|
{
|
||||||
|
variable = rootVariableDeclaration(*variable);
|
||||||
u256 value;
|
u256 value;
|
||||||
if (variable->value()->annotation().type->category() == Type::Category::RationalNumber)
|
if (variable->value()->annotation().type->category() == Type::Category::RationalNumber)
|
||||||
{
|
{
|
||||||
|
@ -0,0 +1,26 @@
|
|||||||
|
contract C {
|
||||||
|
uint constant a = 2;
|
||||||
|
uint constant aa = a;
|
||||||
|
uint constant aaa = aa;
|
||||||
|
bytes2 constant b = 0xabcd;
|
||||||
|
bytes2 constant bb = b;
|
||||||
|
bytes3 constant c = "abc";
|
||||||
|
bytes3 constant cc = c;
|
||||||
|
bytes3 constant ccc = cc;
|
||||||
|
bytes3 constant cccc = ccc;
|
||||||
|
bool constant d = true;
|
||||||
|
bool constant dd = d;
|
||||||
|
address payable constant e = 0x1212121212121212121212121212121212121212;
|
||||||
|
address payable constant ee = e;
|
||||||
|
function f() public pure returns (uint w, bytes2 x, bytes3 y, bool z, address t) {
|
||||||
|
assembly {
|
||||||
|
w := aaa
|
||||||
|
x := bb
|
||||||
|
y := cccc
|
||||||
|
z := dd
|
||||||
|
t := ee
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// f() -> 2, left(0xabcd), left(0x616263), true, 0x1212121212121212121212121212121212121212
|
@ -7,4 +7,4 @@ contract C {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError: (115-116): Only direct number constants are supported by inline assembly.
|
// TypeError: (115-116): Only direct number constants and references to such constants are supported by inline assembly.
|
||||||
|
@ -0,0 +1,11 @@
|
|||||||
|
contract C {
|
||||||
|
bytes32 constant x = keccak256("abc");
|
||||||
|
bytes32 constant y = x;
|
||||||
|
function f() public pure returns (uint t) {
|
||||||
|
assembly {
|
||||||
|
t := y
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// TypeError: (168-169): Only direct number constants and references to such constants are supported by inline assembly.
|
@ -8,4 +8,3 @@ contract C {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// TypeError: (134-135): Only direct number constants are supported by inline assembly.
|
|
||||||
|
Loading…
Reference in New Issue
Block a user