/* 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 . */ /** * Yul dialects for EVM. */ #include #include #include #include #include #include #include #include #include #include #include using namespace std; using namespace dev; using namespace yul; namespace { pair createEVMFunction( string const& _name, dev::eth::Instruction _instruction ) { eth::InstructionInfo info = dev::eth::instructionInfo(_instruction); BuiltinFunctionForEVM f; f.name = YulString{_name}; f.parameters.resize(info.args); f.returns.resize(info.ret); f.sideEffects = EVMDialect::sideEffectsOfInstruction(_instruction); f.isMSize = _instruction == dev::eth::Instruction::MSIZE; f.literalArguments = false; f.instruction = _instruction; f.generateCode = [_instruction]( FunctionCall const&, AbstractAssembly& _assembly, BuiltinContext&, std::function _visitArguments ) { _visitArguments(); _assembly.appendInstruction(_instruction); }; return {f.name, move(f)}; } pair createFunction( string _name, size_t _params, size_t _returns, SideEffects _sideEffects, bool _literalArguments, std::function)> _generateCode ) { YulString name{std::move(_name)}; BuiltinFunctionForEVM f; f.name = name; f.parameters.resize(_params); f.returns.resize(_returns); f.sideEffects = std::move(_sideEffects); f.literalArguments = _literalArguments; f.isMSize = false; f.instruction = {}; f.generateCode = std::move(_generateCode); return {name, f}; } map createBuiltins(langutil::EVMVersion _evmVersion, bool _objectAccess) { map builtins; for (auto const& instr: Parser::instructions()) if ( !dev::eth::isDupInstruction(instr.second) && !dev::eth::isSwapInstruction(instr.second) && instr.second != eth::Instruction::JUMP && instr.second != eth::Instruction::JUMPI && _evmVersion.hasOpcode(instr.second) ) builtins.emplace(createEVMFunction(instr.first, instr.second)); if (_objectAccess) { builtins.emplace(createFunction("datasize", 1, 1, SideEffects{}, true, []( FunctionCall const& _call, AbstractAssembly& _assembly, BuiltinContext& _context, function ) { yulAssert(_context.currentObject, "No object available."); yulAssert(_call.arguments.size() == 1, ""); Expression const& arg = _call.arguments.front(); YulString dataName = boost::get(arg).value; if (_context.currentObject->name == dataName) _assembly.appendAssemblySize(); else { yulAssert( _context.subIDs.count(dataName) != 0, "Could not find assembly object <" + dataName.str() + ">." ); _assembly.appendDataSize(_context.subIDs.at(dataName)); } })); builtins.emplace(createFunction("dataoffset", 1, 1, SideEffects{}, true, []( FunctionCall const& _call, AbstractAssembly& _assembly, BuiltinContext& _context, std::function ) { yulAssert(_context.currentObject, "No object available."); yulAssert(_call.arguments.size() == 1, ""); Expression const& arg = _call.arguments.front(); YulString dataName = boost::get(arg).value; if (_context.currentObject->name == dataName) _assembly.appendConstant(0); else { yulAssert( _context.subIDs.count(dataName) != 0, "Could not find assembly object <" + dataName.str() + ">." ); _assembly.appendDataOffset(_context.subIDs.at(dataName)); } })); builtins.emplace(createFunction( "datacopy", 3, 0, SideEffects{false, false, false, false, true}, false, []( FunctionCall const&, AbstractAssembly& _assembly, BuiltinContext&, std::function _visitArguments ) { _visitArguments(); _assembly.appendInstruction(dev::eth::Instruction::CODECOPY); } )); } return builtins; } } EVMDialect::EVMDialect(AsmFlavour _flavour, bool _objectAccess, langutil::EVMVersion _evmVersion): Dialect{_flavour}, m_objectAccess(_objectAccess), m_evmVersion(_evmVersion), m_functions(createBuiltins(_evmVersion, _objectAccess)) { } BuiltinFunctionForEVM const* EVMDialect::builtin(YulString _name) const { auto it = m_functions.find(_name); if (it != m_functions.end()) return &it->second; else return nullptr; } EVMDialect const& EVMDialect::looseAssemblyForEVM(langutil::EVMVersion _version) { static map> dialects; static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }}; if (!dialects[_version]) dialects[_version] = make_unique(AsmFlavour::Loose, false, _version); return *dialects[_version]; } EVMDialect const& EVMDialect::strictAssemblyForEVM(langutil::EVMVersion _version) { static map> dialects; static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }}; if (!dialects[_version]) dialects[_version] = make_unique(AsmFlavour::Strict, false, _version); return *dialects[_version]; } EVMDialect const& EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion _version) { static map> dialects; static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }}; if (!dialects[_version]) dialects[_version] = make_unique(AsmFlavour::Strict, true, _version); return *dialects[_version]; } EVMDialect const& EVMDialect::yulForEVM(langutil::EVMVersion _version) { static map> dialects; static YulStringRepository::ResetCallback callback{[&] { dialects.clear(); }}; if (!dialects[_version]) dialects[_version] = make_unique(AsmFlavour::Yul, false, _version); return *dialects[_version]; } SideEffects EVMDialect::sideEffectsOfInstruction(eth::Instruction _instruction) { return SideEffects{ eth::SemanticInformation::movable(_instruction), eth::SemanticInformation::sideEffectFree(_instruction), eth::SemanticInformation::sideEffectFreeIfNoMSize(_instruction), eth::SemanticInformation::invalidatesStorage(_instruction), eth::SemanticInformation::invalidatesMemory(_instruction) }; }