/*
	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 .
*/
/**
 * Specific AST walkers that collect semantical facts.
 */
#pragma once
#include 
#include 
namespace yul
{
struct Dialect;
/**
 * Specific AST walker that determines side-effect free-ness and movability of code.
 * Enters into function definitions.
 */
class SideEffectsCollector: public ASTWalker
{
public:
	explicit SideEffectsCollector(Dialect const& _dialect): m_dialect(_dialect) {}
	SideEffectsCollector(Dialect const& _dialect, Expression const& _expression);
	SideEffectsCollector(Dialect const& _dialect, Statement const& _statement);
	SideEffectsCollector(Dialect const& _dialect, Block const& _ast);
	using ASTWalker::operator();
	void operator()(FunctionalInstruction const& _functionalInstruction) override;
	void operator()(FunctionCall const& _functionCall) override;
	bool movable() const { return m_movable; }
	bool sideEffectFree(bool _allowMSizeModification = false) const
	{
		if (_allowMSizeModification)
			return sideEffectFreeIfNoMSize();
		else
			return m_sideEffectFree;
	}
	bool sideEffectFreeIfNoMSize() const { return m_sideEffectFreeIfNoMSize; }
	bool containsMSize() const { return m_containsMSize; }
	bool invalidatesStorage() const { return m_invalidatesStorage; }
	bool invalidatesMemory() const { return m_invalidatesMemory; }
private:
	Dialect const& m_dialect;
	/// Is the current expression movable or not.
	bool m_movable = true;
	/// Is the current expression side-effect free, i.e. can be removed
	/// without changing the semantics.
	bool m_sideEffectFree = true;
	/// Is the current expression side-effect free up to msize, i.e. can be removed
	/// without changing the semantics except for the value returned by the msize instruction.
	bool m_sideEffectFreeIfNoMSize = true;
	/// Does the current code contain the MSize operation?
	/// Note that this is a purely syntactic property meaning that even if this is false,
	/// the code can still contain calls to functions that contain the msize instruction.
	bool m_containsMSize = false;
	/// If false, storage is guaranteed to be unchanged by the code under all
	/// circumstances.
	bool m_invalidatesStorage = false;
	bool m_invalidatesMemory = false;
};
/**
 * Specific AST walker that determines whether an expression is movable
 * and collects the referenced variables.
 * Can only be used on expressions.
 */
class MovableChecker: public SideEffectsCollector
{
public:
	explicit MovableChecker(Dialect const& _dialect): SideEffectsCollector(_dialect) {}
	MovableChecker(Dialect const& _dialect, Expression const& _expression);
	void operator()(Identifier const& _identifier) override;
	/// Disallow visiting anything apart from Expressions (this throws).
	void visit(Statement const&) override;
	using ASTWalker::visit;
	std::set const& referencedVariables() const { return m_variableReferences; }
private:
	/// Which variables the current expression references.
	std::set m_variableReferences;
};
/**
 * Helper class to find "irregular" control flow.
 * This includes termination, break and continue.
 */
class TerminationFinder
{
public:
	enum class ControlFlow { FlowOut, Break, Continue, Terminate };
	TerminationFinder(Dialect const& _dialect): m_dialect(_dialect) {}
	/// @returns the index of the first statement in the provided sequence
	/// that is an unconditional ``break``, ``continue`` or a
	/// call to a terminating builtin function.
	/// If control flow can continue at the end of the list,
	/// returns `FlowOut` and ``size_t(-1)``.
	/// The function might return ``FlowOut`` even though control
	/// flow cannot actually continue.
	std::pair firstUnconditionalControlFlowChange(
		std::vector const& _statements
	);
	/// @returns the control flow type of the given statement.
	/// This function could return FlowOut even if control flow never continues.
	ControlFlow controlFlowKind(Statement const& _statement);
	/// @returns true if the expression statement is a direct
	/// call to a builtin terminating function like
	/// ``stop``, ``revert`` or ``return``.
	bool isTerminatingBuiltin(ExpressionStatement const& _exprStmnt);
private:
	Dialect const& m_dialect;
};
}