Merge pull request #9549 from ethereum/yul-reserved

[BREAKING] Disallow using reserved identifiers in Yul
This commit is contained in:
chriseth 2020-12-10 18:49:44 +01:00 committed by GitHub
commit ef3bcbda97
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
15 changed files with 122 additions and 8 deletions

View File

@ -26,6 +26,7 @@ Breaking Changes:
* Type System: Support ``address(...).codehash`` to retrieve the codehash of an account.
* Type System: Unary negation can only be used on signed integers, not on unsigned integers.
* View Pure Checker: Mark ``chainid`` as view.
* Yul: Disallow the use of reserved identifiers, such as EVM instructions, even if they are not available in the given dialect / EVM version.
Language Features:
* Super constructors can now be called using the member notation e.g. ``M.C(123)``.

View File

@ -570,6 +570,12 @@ void AsmAnalyzer::expectValidIdentifier(YulString _identifier, SourceLocation co
"\"" + _identifier.str() + "\" is not a valid identifier (contains consecutive dots)."
);
if (m_dialect.reservedIdentifier(_identifier))
m_errorReporter.declarationError(
5017_error,
_location,
"The identifier \"" + _identifier.str() + "\" is reserved and can not be used."
);
}
void AsmAnalyzer::expectValidType(YulString _type, SourceLocation const& _location)

View File

@ -68,6 +68,9 @@ struct Dialect: boost::noncopyable
/// @returns the builtin function of the given name or a nullptr if it is not a builtin function.
virtual BuiltinFunction const* builtin(YulString /*_name*/) const { return nullptr; }
/// @returns true if the identifier is reserved. This includes the builtins too.
virtual bool reservedIdentifier(YulString _name) const { return builtin(_name) != nullptr; }
virtual BuiltinFunction const* discardFunction(YulString /* _type */) const { return nullptr; }
virtual BuiltinFunction const* equalityFunction(YulString /* _type */) const { return nullptr; }
virtual BuiltinFunction const* booleanNegationFunction() const { return nullptr; }

View File

@ -109,6 +109,26 @@ pair<YulString, BuiltinFunctionForEVM> createFunction(
return {name, f};
}
set<YulString> createReservedIdentifiers()
{
set<YulString> reserved;
for (auto const& instr: evmasm::c_instructions)
{
string name = instr.first;
transform(name.begin(), name.end(), name.begin(), [](unsigned char _c) { return tolower(_c); });
reserved.emplace(name);
}
reserved += vector<YulString>{
"linkersymbol"_yulstring,
"datasize"_yulstring,
"dataoffset"_yulstring,
"datacopy"_yulstring,
"setimmutable"_yulstring,
"loadimmutable"_yulstring,
};
return reserved;
}
map<YulString, BuiltinFunctionForEVM> createBuiltins(langutil::EVMVersion _evmVersion, bool _objectAccess)
{
map<YulString, BuiltinFunctionForEVM> builtins;
@ -266,7 +286,8 @@ map<YulString, BuiltinFunctionForEVM> createBuiltins(langutil::EVMVersion _evmVe
EVMDialect::EVMDialect(langutil::EVMVersion _evmVersion, bool _objectAccess):
m_objectAccess(_objectAccess),
m_evmVersion(_evmVersion),
m_functions(createBuiltins(_evmVersion, _objectAccess))
m_functions(createBuiltins(_evmVersion, _objectAccess)),
m_reserved(createReservedIdentifiers())
{
}
@ -279,6 +300,11 @@ BuiltinFunctionForEVM const* EVMDialect::builtin(YulString _name) const
return nullptr;
}
bool EVMDialect::reservedIdentifier(YulString _name) const
{
return m_reserved.count(_name) != 0;
}
EVMDialect const& EVMDialect::strictAssemblyForEVM(langutil::EVMVersion _version)
{
static map<langutil::EVMVersion, unique_ptr<EVMDialect const>> dialects;

View File

@ -28,6 +28,7 @@
#include <liblangutil/EVMVersion.h>
#include <map>
#include <set>
namespace solidity::yul
{
@ -70,6 +71,9 @@ struct EVMDialect: public Dialect
/// @returns the builtin function of the given name or a nullptr if it is not a builtin function.
BuiltinFunctionForEVM const* builtin(YulString _name) const override;
/// @returns true if the identifier is reserved. This includes the builtins too.
bool reservedIdentifier(YulString _name) const override;
BuiltinFunctionForEVM const* discardFunction(YulString /*_type*/) const override { return builtin("pop"_yulstring); }
BuiltinFunctionForEVM const* equalityFunction(YulString /*_type*/) const override { return builtin("eq"_yulstring); }
BuiltinFunctionForEVM const* booleanNegationFunction() const override { return builtin("iszero"_yulstring); }
@ -91,6 +95,7 @@ protected:
bool const m_objectAccess;
langutil::EVMVersion const m_evmVersion;
std::map<YulString, BuiltinFunctionForEVM> m_functions;
std::set<YulString> m_reserved;
};
/**

View File

@ -45,5 +45,5 @@ void yul::removeEmptyBlocks(Block& _block)
bool yul::isRestrictedIdentifier(Dialect const& _dialect, YulString const& _identifier)
{
return _identifier.empty() || TokenTraits::isYulKeyword(_identifier.str()) || _dialect.builtin(_identifier);
return _identifier.empty() || TokenTraits::isYulKeyword(_identifier.str()) || _dialect.reservedIdentifier(_identifier);
}

View File

@ -359,12 +359,6 @@ BOOST_AUTO_TEST_CASE(shift_constantinople_warning)
CHECK_PARSE_WARNING("{ sar(10, 32) }", TypeError, "The \"sar\" instruction is only available for Constantinople-compatible VMs");
}
BOOST_AUTO_TEST_CASE(jump_error)
{
CHECK_PARSE_WARNING("{ jump(44) }", DeclarationError, "Function not found.");
CHECK_PARSE_WARNING("{ jumpi(44, 2) }", DeclarationError, "Function not found.");
}
BOOST_AUTO_TEST_SUITE_END() // }}}
BOOST_AUTO_TEST_SUITE_END()

View File

@ -8,3 +8,4 @@ contract C {
}
}
// ----
// DeclarationError 5017: (63-90): The identifier "linkersymbol" is reserved and can not be used.

View File

@ -0,0 +1,13 @@
contract C {
function f() public pure {
assembly {
let linkersymbol := 1
let datacopy := 1
let swap16 := 1
}
}
}
// ----
// DeclarationError 5017: (67-79): The identifier "linkersymbol" is reserved and can not be used.
// DeclarationError 5017: (95-103): The identifier "datacopy" is reserved and can not be used.
// DeclarationError 5017: (119-125): The identifier "swap16" is reserved and can not be used.

View File

@ -0,0 +1,16 @@
contract C {
function f() public pure {
assembly {
let shl := 1
}
assembly {
pop(shl(1, 2))
}
}
}
// ====
// EVMVersion: =byzantium
// ----
// DeclarationError 5017: (67-70): The identifier "shl" is reserved and can not be used.
// TypeError 6612: (107-110): The "shl" instruction is only available for Constantinople-compatible VMs (you are currently compiling for "byzantium").
// TypeError 3950: (107-116): Expected expression to evaluate to one value, but got 0 values instead.

View File

@ -0,0 +1,14 @@
contract C {
function f() public pure {
assembly {
let shl := 1
}
assembly {
pop(shl(1, 2))
}
}
}
// ====
// EVMVersion: >=constantinople
// ----
// ParserError 5568: (67-70): Cannot use builtin function name "shl" as identifier name.

View File

@ -0,0 +1,32 @@
{
function g_(x) -> z_ {
if calldataload(1) { z_ := g_(x) }
sstore(z_, calldataload(add(x, 1)))
}
function datasize_(x) -> linkersymbol_ {
if calldataload(0) { linkersymbol_ := datasize_(x) }
sstore(linkersymbol_, calldataload(linkersymbol_))
}
let dataoffset_ := datasize_(7)
let x_ := g_(9)
sstore(dataoffset_, x_)
}
// ----
// step: fullSuite
//
// {
// {
// let dataoffset_ := datasize_(7)
// sstore(dataoffset_, g(9))
// }
// function g(x) -> z
// {
// if calldataload(1) { z := g(x) }
// sstore(z, calldataload(add(x, 1)))
// }
// function datasize_(x) -> linkersymbol_
// {
// if calldataload(linkersymbol_) { linkersymbol_ := datasize_(x) }
// sstore(linkersymbol_, calldataload(linkersymbol_))
// }
// }

View File

@ -1,6 +1,8 @@
{
let addr:u256 := linkersymbol("contract/library.sol:L")
function linkersymbol(x) {}
}
// ====
// dialect: evmTyped
// ----
// ParserError 5568: (75-87): Cannot use builtin function name "linkersymbol" as identifier name.

View File

@ -6,3 +6,4 @@
let s := ""
datacopy(x, "11", s)
}
// ----