/*
	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;
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(dev::eth::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);
}