mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			144 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			144 lines
		
	
	
		
			4.2 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
| 	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/>.
 | |
| */
 | |
| /**
 | |
|  * Yul dialects for EVM.
 | |
|  */
 | |
| 
 | |
| #include <libyul/backends/evm/EVMDialect.h>
 | |
| 
 | |
| #include <libyul/AsmAnalysisInfo.h>
 | |
| #include <libyul/AsmData.h>
 | |
| #include <libyul/Object.h>
 | |
| #include <libyul/backends/evm/AbstractAssembly.h>
 | |
| 
 | |
| #include <liblangutil/Exceptions.h>
 | |
| 
 | |
| #include <libyul/Exceptions.h>
 | |
| 
 | |
| #include <boost/range/adaptor/reversed.hpp>
 | |
| 
 | |
| 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<void()>
 | |
| 	) {
 | |
| 		yulAssert(m_currentObject, "No object available.");
 | |
| 		yulAssert(_call.arguments.size() == 1, "");
 | |
| 		Expression const& arg = _call.arguments.front();
 | |
| 		YulString dataName = boost::get<Literal>(arg).value;
 | |
| 		if (m_currentObject->name == dataName)
 | |
| 			_assembly.appendAssemblySize();
 | |
| 		else
 | |
| 			_assembly.appendDataSize(m_subIDs.at(dataName));
 | |
| 	});
 | |
| 	addFunction("dataoffset", 1, 1, true, true, [this](
 | |
| 		FunctionCall const& _call,
 | |
| 		AbstractAssembly& _assembly,
 | |
| 		std::function<void()>
 | |
| 	) {
 | |
| 		yulAssert(m_currentObject, "No object available.");
 | |
| 		yulAssert(_call.arguments.size() == 1, "");
 | |
| 		Expression const& arg = _call.arguments.front();
 | |
| 		YulString dataName = boost::get<Literal>(arg).value;
 | |
| 		if (m_currentObject->name == dataName)
 | |
| 			_assembly.appendConstant(0);
 | |
| 		else
 | |
| 			_assembly.appendDataOffset(m_subIDs.at(dataName));
 | |
| 	});
 | |
| 	addFunction("datacopy", 3, 0, false, false, [](
 | |
| 		FunctionCall const&,
 | |
| 		AbstractAssembly& _assembly,
 | |
| 		std::function<void()> _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> EVMDialect::looseAssemblyForEVM(langutil::EVMVersion _version)
 | |
| {
 | |
| 	return make_shared<EVMDialect>(AsmFlavour::Loose, false, _version);
 | |
| }
 | |
| 
 | |
| shared_ptr<EVMDialect> EVMDialect::strictAssemblyForEVM(langutil::EVMVersion _version)
 | |
| {
 | |
| 	return make_shared<EVMDialect>(AsmFlavour::Strict, false, _version);
 | |
| }
 | |
| 
 | |
| shared_ptr<EVMDialect> EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion _version)
 | |
| {
 | |
| 	return make_shared<EVMDialect>(AsmFlavour::Strict, true, _version);
 | |
| }
 | |
| 
 | |
| shared_ptr<yul::EVMDialect> EVMDialect::yulForEVM(langutil::EVMVersion _version)
 | |
| {
 | |
| 	return make_shared<EVMDialect>(AsmFlavour::Yul, false, _version);
 | |
| }
 | |
| 
 | |
| void EVMDialect::setSubIDs(map<YulString, AbstractAssembly::SubID> _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<void(FunctionCall const&, AbstractAssembly&, std::function<void()>)> _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);
 | |
| }
 |