/* 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 using namespace std; using namespace dev; using namespace yul; using namespace dev::solidity; EVMDialect::EVMDialect(AsmFlavour _flavour, bool _objectAccess, langutil::EVMVersion _evmVersion): Dialect{_flavour}, m_objectAccess(_objectAccess), m_evmVersion(_evmVersion) { // The EVM instructions will be moved to builtins at some point. if (!m_objectAccess) return; addFunction("datasize", 1, 1, true, true, [this]( FunctionCall const& _call, AbstractAssembly& _assembly, std::function ) { yulAssert(m_currentObject, "No object available."); yulAssert(_call.arguments.size() == 1, ""); Expression const& arg = _call.arguments.front(); YulString dataName = boost::get(arg).value; if (m_currentObject->name == dataName) _assembly.appendAssemblySize(); else { yulAssert(m_subIDs.count(dataName) != 0, "Could not find assembly object <" + dataName.str() + ">."); _assembly.appendDataSize(m_subIDs.at(dataName)); } }); addFunction("dataoffset", 1, 1, true, true, [this]( FunctionCall const& _call, AbstractAssembly& _assembly, std::function ) { yulAssert(m_currentObject, "No object available."); yulAssert(_call.arguments.size() == 1, ""); Expression const& arg = _call.arguments.front(); YulString dataName = boost::get(arg).value; if (m_currentObject->name == dataName) _assembly.appendConstant(0); else { yulAssert(m_subIDs.count(dataName) != 0, "Could not find assembly object <" + dataName.str() + ">."); _assembly.appendDataOffset(m_subIDs.at(dataName)); } }); addFunction("datacopy", 3, 0, false, false, []( FunctionCall const&, AbstractAssembly& _assembly, std::function _visitArguments ) { _visitArguments(); _assembly.appendInstruction(solidity::Instruction::CODECOPY); }); } BuiltinFunctionForEVM const* EVMDialect::builtin(YulString _name) const { auto it = m_functions.find(_name); if (it != m_functions.end()) return &it->second; else return nullptr; } shared_ptr EVMDialect::looseAssemblyForEVM(langutil::EVMVersion _version) { return make_shared(AsmFlavour::Loose, false, _version); } shared_ptr EVMDialect::strictAssemblyForEVM(langutil::EVMVersion _version) { return make_shared(AsmFlavour::Strict, false, _version); } shared_ptr EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion _version) { return make_shared(AsmFlavour::Strict, true, _version); } shared_ptr EVMDialect::yulForEVM(langutil::EVMVersion _version) { return make_shared(AsmFlavour::Yul, false, _version); } void EVMDialect::setSubIDs(map _subIDs) { yulAssert(m_objectAccess, "Sub IDs set with dialect that does not support object access."); m_subIDs = std::move(_subIDs); } void EVMDialect::setCurrentObject(Object const* _object) { yulAssert(m_objectAccess, "Current object set with dialect that does not support object access."); m_currentObject = _object; } void EVMDialect::addFunction( string _name, size_t _params, size_t _returns, bool _movable, bool _literalArguments, std::function)> _generateCode ) { YulString name{std::move(_name)}; BuiltinFunctionForEVM& f = m_functions[name]; f.name = name; f.parameters.resize(_params); f.returns.resize(_returns); f.movable = _movable; f.literalArguments = _literalArguments; f.generateCode = std::move(_generateCode); }