/*
	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 .
*/
// SPDX-License-Identifier: GPL-3.0
/**
 * Creates an independent copy of an AST, renaming identifiers to be unique.
 */
#include 
#include 
#include 
using namespace std;
using namespace solidity;
using namespace solidity::yul;
using namespace solidity::util;
Statement ASTCopier::operator()(ExpressionStatement const& _statement)
{
	return ExpressionStatement{ _statement.debugData, translate(_statement.expression) };
}
Statement ASTCopier::operator()(VariableDeclaration const& _varDecl)
{
	return VariableDeclaration{
		_varDecl.debugData,
		translateVector(_varDecl.variables),
		translate(_varDecl.value)
	};
}
Statement ASTCopier::operator()(Assignment const& _assignment)
{
	return Assignment{
		_assignment.debugData,
		translateVector(_assignment.variableNames),
		translate(_assignment.value)
	};
}
Expression ASTCopier::operator()(FunctionCall const& _call)
{
	return FunctionCall{
		_call.debugData,
		translate(_call.functionName),
		translateVector(_call.arguments)
	};
}
Expression ASTCopier::operator()(Identifier const& _identifier)
{
	return translate(_identifier);
}
Expression ASTCopier::operator()(Literal const& _literal)
{
	return translate(_literal);
}
Statement ASTCopier::operator()(If const& _if)
{
	return If{_if.debugData, translate(_if.condition), translate(_if.body)};
}
Statement ASTCopier::operator()(Switch const& _switch)
{
	return Switch{_switch.debugData, translate(_switch.expression), translateVector(_switch.cases)};
}
Statement ASTCopier::operator()(FunctionDefinition const& _function)
{
	YulString translatedName = translateIdentifier(_function.name);
	enterFunction(_function);
	ScopeGuard g([&]() { this->leaveFunction(_function); });
	return FunctionDefinition{
		_function.debugData,
		translatedName,
		translateVector(_function.parameters),
		translateVector(_function.returnVariables),
		translate(_function.body)
	};
}
Statement ASTCopier::operator()(ForLoop const& _forLoop)
{
	enterScope(_forLoop.pre);
	ScopeGuard g([&]() { this->leaveScope(_forLoop.pre); });
	return ForLoop{
		_forLoop.debugData,
		translate(_forLoop.pre),
		translate(_forLoop.condition),
		translate(_forLoop.post),
		translate(_forLoop.body)
	};
}
Statement ASTCopier::operator()(Break const& _break)
{
	return Break{ _break };
}
Statement ASTCopier::operator()(Continue const& _continue)
{
	return Continue{ _continue };
}
Statement ASTCopier::operator()(Leave const& _leaveStatement)
{
	return Leave{_leaveStatement};
}
Statement ASTCopier::operator ()(Block const& _block)
{
	return translate(_block);
}
Expression ASTCopier::translate(Expression const& _expression)
{
	return std::visit(static_cast(*this), _expression);
}
Statement ASTCopier::translate(Statement const& _statement)
{
	return std::visit(static_cast(*this), _statement);
}
Block ASTCopier::translate(Block const& _block)
{
	enterScope(_block);
	ScopeGuard g([&]() { this->leaveScope(_block); });
	return Block{_block.debugData, translateVector(_block.statements)};
}
Case ASTCopier::translate(Case const& _case)
{
	return Case{_case.debugData, translate(_case.value), translate(_case.body)};
}
Identifier ASTCopier::translate(Identifier const& _identifier)
{
	return Identifier{_identifier.debugData, translateIdentifier(_identifier.name)};
}
Literal ASTCopier::translate(Literal const& _literal)
{
	return _literal;
}
TypedName ASTCopier::translate(TypedName const& _typedName)
{
	return TypedName{_typedName.debugData, translateIdentifier(_typedName.name), _typedName.type};
}
YulString FunctionCopier::translateIdentifier(YulString _name)
{
	if (m_translations.count(_name))
		return m_translations.at(_name);
	return _name;
}