/*
	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 .
*/
#include 
#include 
#include 
#include 
#include 
#include 
#include 
#include 
using namespace std;
using namespace solidity;
using namespace solidity::langutil;
using namespace solidity::frontend;
bool PostTypeChecker::check(ASTNode const& _astRoot)
{
	_astRoot.accept(*this);
	return Error::containsOnlyWarnings(m_errorReporter.errors());
}
bool PostTypeChecker::visit(ContractDefinition const& _contractDefinition)
{
	return callVisit(_contractDefinition);
}
void PostTypeChecker::endVisit(ContractDefinition const& _contractDefinition)
{
	callEndVisit(_contractDefinition);
}
void PostTypeChecker::endVisit(OverrideSpecifier const& _overrideSpecifier)
{
	callEndVisit(_overrideSpecifier);
}
bool PostTypeChecker::visit(VariableDeclaration const& _variable)
{
	return callVisit(_variable);
}
void PostTypeChecker::endVisit(VariableDeclaration const& _variable)
{
	callEndVisit(_variable);
}
bool PostTypeChecker::visit(EmitStatement const& _emit)
{
	return callVisit(_emit);
}
void PostTypeChecker::endVisit(EmitStatement const& _emit)
{
	callEndVisit(_emit);
}
bool PostTypeChecker::visit(FunctionCall const& _functionCall)
{
	return callVisit(_functionCall);
}
bool PostTypeChecker::visit(Identifier const& _identifier)
{
	return callVisit(_identifier);
}
bool PostTypeChecker::visit(StructDefinition const& _struct)
{
	return callVisit(_struct);
}
void PostTypeChecker::endVisit(StructDefinition const& _struct)
{
	callEndVisit(_struct);
}
bool PostTypeChecker::visit(ModifierInvocation const& _modifierInvocation)
{
	return callVisit(_modifierInvocation);
}
void PostTypeChecker::endVisit(ModifierInvocation const& _modifierInvocation)
{
	callEndVisit(_modifierInvocation);
}
namespace
{
struct ConstStateVarCircularReferenceChecker: public PostTypeChecker::Checker
{
	ConstStateVarCircularReferenceChecker(ErrorReporter& _errorReporter):
		Checker(_errorReporter) {}
	bool visit(ContractDefinition const&) override
	{
		solAssert(!m_currentConstVariable, "");
		solAssert(m_constVariableDependencies.empty(), "");
		return true;
	}
	void endVisit(ContractDefinition const&) override
	{
		solAssert(!m_currentConstVariable, "");
		for (auto declaration: m_constVariables)
			if (auto identifier = findCycle(*declaration))
				m_errorReporter.typeError(
					declaration->location(),
					"The value of the constant " + declaration->name() +
					" has a cyclic dependency via " + identifier->name() + "."
				);
		m_constVariables.clear();
		m_constVariableDependencies.clear();
	}
	bool visit(VariableDeclaration const& _variable) override
	{
		if (_variable.isConstant())
		{
			solAssert(!m_currentConstVariable, "");
			m_currentConstVariable = &_variable;
			m_constVariables.push_back(&_variable);
		}
		return true;
	}
	void endVisit(VariableDeclaration const& _variable) override
	{
		if (_variable.isConstant())
		{
			solAssert(m_currentConstVariable == &_variable, "");
			m_currentConstVariable = nullptr;
		}
	}
	bool visit(Identifier const& _identifier) override
	{
		if (m_currentConstVariable)
			if (auto var = dynamic_cast(_identifier.annotation().referencedDeclaration))
				if (var->isConstant())
					m_constVariableDependencies[m_currentConstVariable].insert(var);
		return true;
	}
	VariableDeclaration const* findCycle(VariableDeclaration const& _startingFrom)
	{
		auto visitor = [&](VariableDeclaration const& _variable, util::CycleDetector& _cycleDetector, size_t _depth)
		{
			if (_depth >= 256)
				m_errorReporter.fatalDeclarationError(_variable.location(), "Variable definition exhausting cyclic dependency validator.");
			// Iterating through the dependencies needs to be deterministic and thus cannot
			// depend on the memory layout.
			// Because of that, we sort by AST node id.
			vector dependencies(
				m_constVariableDependencies[&_variable].begin(),
				m_constVariableDependencies[&_variable].end()
			);
			sort(dependencies.begin(), dependencies.end(), [](VariableDeclaration const* _a, VariableDeclaration const* _b) -> bool
			{
				return _a->id() < _b->id();
			});
			for (auto v: dependencies)
				if (_cycleDetector.run(*v))
					return;
		};
		return util::CycleDetector(visitor).run(_startingFrom);
	}
private:
	VariableDeclaration const* m_currentConstVariable = nullptr;
	std::map> m_constVariableDependencies;
	std::vector m_constVariables; ///< Required for determinism.
};
struct OverrideSpecifierChecker: public PostTypeChecker::Checker
{
	OverrideSpecifierChecker(ErrorReporter& _errorReporter):
		Checker(_errorReporter) {}
	void endVisit(OverrideSpecifier const& _overrideSpecifier) override
	{
		for (ASTPointer const& override: _overrideSpecifier.overrides())
		{
			Declaration const* decl  = override->annotation().referencedDeclaration;
			solAssert(decl, "Expected declaration to be resolved.");
			if (dynamic_cast(decl))
				continue;
			TypeType const* actualTypeType = dynamic_cast(decl->type());
			m_errorReporter.typeError(
				override->location(),
				"Expected contract but got " +
				actualTypeType->actualType()->toString(true) +
				"."
			);
		}
	}
};
struct ModifierContextChecker: public PostTypeChecker::Checker
{
	ModifierContextChecker(ErrorReporter& _errorReporter):
		Checker(_errorReporter) {}
	bool visit(ModifierInvocation const&) override
	{
		m_insideModifierInvocation = true;
		return true;
	}
	void endVisit(ModifierInvocation const&) override
	{
		m_insideModifierInvocation = false;
	}
	bool visit(Identifier const& _identifier) override
	{
		if (m_insideModifierInvocation)
			return true;
		if (ModifierType const* type = dynamic_cast(_identifier.annotation().type))
		{
			m_errorReporter.typeError(
				_identifier.location(),
				"Modifier can only be referenced in function headers."
			);
		}
		return false;
	}
private:
	/// Flag indicating whether we are currently inside the invocation of a modifier
	bool m_insideModifierInvocation = false;
};
struct EventOutsideEmitChecker: public PostTypeChecker::Checker
{
	EventOutsideEmitChecker(ErrorReporter& _errorReporter):
		Checker(_errorReporter) {}
	bool visit(EmitStatement const&) override
	{
		m_insideEmitStatement = true;
		return true;
	}
	void endVisit(EmitStatement const&) override
	{
		m_insideEmitStatement = true;
	}
	bool visit(FunctionCall const& _functionCall) override
	{
		if (_functionCall.annotation().kind != FunctionCallKind::FunctionCall)
			return true;
		if (FunctionTypePointer const functionType = dynamic_cast(_functionCall.expression().annotation().type))
			// Check for event outside of emit statement
			if (!m_insideEmitStatement && functionType->kind() == FunctionType::Kind::Event)
				m_errorReporter.typeError(
					_functionCall.location(),
					"Event invocations have to be prefixed by \"emit\"."
				);
		return true;
	}
private:
	/// Flag indicating whether we are currently inside an EmitStatement.
	bool m_insideEmitStatement = false;
};
struct NoVariablesInInterfaceChecker: public PostTypeChecker::Checker
{
	NoVariablesInInterfaceChecker(ErrorReporter& _errorReporter):
		Checker(_errorReporter)
	{}
	bool visit(VariableDeclaration const& _variable) override
	{
		// Forbid any variable declarations inside interfaces unless they are part of
		// * a function's input/output parameters,
		// * or inside of a struct definition.
		if (
			m_scope && m_scope->isInterface()
			&& !_variable.isCallableOrCatchParameter()
			&& !m_insideStruct
		)
			m_errorReporter.typeError(_variable.location(), "Variables cannot be declared in interfaces.");
		return true;
	}
	bool visit(ContractDefinition const& _contract) override
	{
		m_scope = &_contract;
		return true;
	}
	void endVisit(ContractDefinition const&) override
	{
		m_scope = nullptr;
	}
	bool visit(StructDefinition const&) override
	{
		solAssert(m_insideStruct >= 0, "");
		m_insideStruct++;
		return true;
	}
	void endVisit(StructDefinition const&) override
	{
		m_insideStruct--;
		solAssert(m_insideStruct >= 0, "");
	}
private:
	ContractDefinition const* m_scope = nullptr;
	/// Flag indicating whether we are currently inside a StructDefinition.
	int m_insideStruct = 0;
};
}
PostTypeChecker::PostTypeChecker(langutil::ErrorReporter& _errorReporter): m_errorReporter(_errorReporter)
{
	m_checkers.push_back(make_shared(_errorReporter));
	m_checkers.push_back(make_shared(_errorReporter));
	m_checkers.push_back(make_shared(_errorReporter));
	m_checkers.push_back(make_shared(_errorReporter));
	m_checkers.push_back(make_shared(_errorReporter));
}