mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			637 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			637 lines
		
	
	
		
			12 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/>.
 | |
| */
 | |
| /**
 | |
|  * @author Christian <c@ethdev.com>
 | |
|  * @date 2014
 | |
|  * Pretty-printer for the abstract syntax tree (the "pretty" is arguable), used for debugging.
 | |
|  */
 | |
| 
 | |
| #include <libsolidity/ast/ASTPrinter.h>
 | |
| 
 | |
| #include <libsolidity/ast/AST.h>
 | |
| #include <boost/algorithm/string/join.hpp>
 | |
| #include <json/json.h>
 | |
| 
 | |
| using namespace std;
 | |
| using namespace langutil;
 | |
| 
 | |
| namespace dev
 | |
| {
 | |
| namespace solidity
 | |
| {
 | |
| 
 | |
| ASTPrinter::ASTPrinter(
 | |
| 	ASTNode const& _ast,
 | |
| 	string const& _source,
 | |
| 	GasEstimator::ASTGasConsumption const& _gasCosts
 | |
| ): m_indentation(0), m_source(_source), m_ast(&_ast), m_gasCosts(_gasCosts)
 | |
| {
 | |
| }
 | |
| 
 | |
| void ASTPrinter::print(ostream& _stream)
 | |
| {
 | |
| 	m_ostream = &_stream;
 | |
| 	m_ast->accept(*this);
 | |
| 	m_ostream = nullptr;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool ASTPrinter::visit(PragmaDirective const& _node)
 | |
| {
 | |
| 	writeLine("PragmaDirective");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ImportDirective const& _node)
 | |
| {
 | |
| 	writeLine("ImportDirective \"" + _node.path() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ContractDefinition const& _node)
 | |
| {
 | |
| 	writeLine("ContractDefinition \"" + _node.name() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(InheritanceSpecifier const& _node)
 | |
| {
 | |
| 	writeLine("InheritanceSpecifier");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(UsingForDirective const& _node)
 | |
| {
 | |
| 	writeLine("UsingForDirective");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(StructDefinition const& _node)
 | |
| {
 | |
| 	writeLine("StructDefinition \"" + _node.name() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(EnumDefinition const& _node)
 | |
| {
 | |
| 	writeLine("EnumDefinition \"" + _node.name() + "\"");
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(EnumValue const& _node)
 | |
| {
 | |
| 	writeLine("EnumValue \"" + _node.name() + "\"");
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ParameterList const& _node)
 | |
| {
 | |
| 	writeLine("ParameterList");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(FunctionDefinition const& _node)
 | |
| {
 | |
| 	writeLine("FunctionDefinition \"" + _node.name() + "\"" +
 | |
| 			  (_node.isPublic() ? " - public" : "") +
 | |
| 			  (_node.stateMutability() == StateMutability::View ? " - const" : ""));
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(VariableDeclaration const& _node)
 | |
| {
 | |
| 	writeLine("VariableDeclaration \"" + _node.name() + "\"");
 | |
| 	*m_ostream << indentation() << (
 | |
| 		_node.annotation().type ?
 | |
| 		string("   Type: ") + _node.annotation().type->toString() :
 | |
| 		string("   Type unknown.")
 | |
| 	) << "\n";
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ModifierDefinition const& _node)
 | |
| {
 | |
| 	writeLine("ModifierDefinition \"" + _node.name() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ModifierInvocation const& _node)
 | |
| {
 | |
| 	writeLine("ModifierInvocation \"" + _node.name()->name() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(EventDefinition const& _node)
 | |
| {
 | |
| 	writeLine("EventDefinition \"" + _node.name() + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ElementaryTypeName const& _node)
 | |
| {
 | |
| 	writeLine(string("ElementaryTypeName ") + _node.typeName().toString());
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(UserDefinedTypeName const& _node)
 | |
| {
 | |
| 	writeLine("UserDefinedTypeName \"" + boost::algorithm::join(_node.namePath(), ".") + "\"");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(FunctionTypeName const& _node)
 | |
| {
 | |
| 	writeLine("FunctionTypeName");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Mapping const& _node)
 | |
| {
 | |
| 	writeLine("Mapping");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ArrayTypeName const& _node)
 | |
| {
 | |
| 	writeLine("ArrayTypeName");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(InlineAssembly const& _node)
 | |
| {
 | |
| 	writeLine("InlineAssembly");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Block const& _node)
 | |
| {
 | |
| 	writeLine("Block");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(PlaceholderStatement const& _node)
 | |
| {
 | |
| 	writeLine("PlaceholderStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(IfStatement const& _node)
 | |
| {
 | |
| 	writeLine("IfStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(WhileStatement const& _node)
 | |
| {
 | |
| 	writeLine(_node.isDoWhile() ? "DoWhileStatement" : "WhileStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ForStatement const& _node)
 | |
| {
 | |
| 	writeLine("ForStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Continue const& _node)
 | |
| {
 | |
| 	writeLine("Continue");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Break const& _node)
 | |
| {
 | |
| 	writeLine("Break");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Return const& _node)
 | |
| {
 | |
| 	writeLine("Return");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Throw const& _node)
 | |
| {
 | |
| 	writeLine("Throw");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(EmitStatement const& _node)
 | |
| {
 | |
| 	writeLine("EmitStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(VariableDeclarationStatement const& _node)
 | |
| {
 | |
| 	writeLine("VariableDeclarationStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ExpressionStatement const& _node)
 | |
| {
 | |
| 	writeLine("ExpressionStatement");
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Conditional const& _node)
 | |
| {
 | |
| 	writeLine("Conditional");
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Assignment const& _node)
 | |
| {
 | |
| 	writeLine(string("Assignment using operator ") + TokenTraits::toString(_node.assignmentOperator()));
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(TupleExpression const& _node)
 | |
| {
 | |
| 	writeLine(string("TupleExpression"));
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(UnaryOperation const& _node)
 | |
| {
 | |
| 	writeLine(string("UnaryOperation (") + (_node.isPrefixOperation() ? "prefix" : "postfix") +
 | |
| 			  ") " + TokenTraits::toString(_node.getOperator()));
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(BinaryOperation const& _node)
 | |
| {
 | |
| 	writeLine(string("BinaryOperation using operator ") + TokenTraits::toString(_node.getOperator()));
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(FunctionCall const& _node)
 | |
| {
 | |
| 	writeLine("FunctionCall");
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(NewExpression const& _node)
 | |
| {
 | |
| 	writeLine("NewExpression");
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(MemberAccess const& _node)
 | |
| {
 | |
| 	writeLine("MemberAccess to member " + _node.memberName());
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(IndexAccess const& _node)
 | |
| {
 | |
| 	writeLine("IndexAccess");
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Identifier const& _node)
 | |
| {
 | |
| 	writeLine(string("Identifier ") + _node.name());
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(ElementaryTypeNameExpression const& _node)
 | |
| {
 | |
| 	writeLine(string("ElementaryTypeNameExpression ") + _node.typeName().toString());
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| bool ASTPrinter::visit(Literal const& _node)
 | |
| {
 | |
| 	char const* tokenString = TokenTraits::toString(_node.token());
 | |
| 	if (!tokenString)
 | |
| 		tokenString = "[no token]";
 | |
| 	writeLine(string("Literal, token: ") + tokenString + " value: " + _node.value());
 | |
| 	printType(_node);
 | |
| 	printSourcePart(_node);
 | |
| 	return goDeeper();
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(PragmaDirective const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ImportDirective const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ContractDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(InheritanceSpecifier const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(UsingForDirective const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(StructDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(EnumDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(EnumValue const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ParameterList const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(FunctionDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(VariableDeclaration const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ModifierDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ModifierInvocation const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(EventDefinition const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ElementaryTypeName const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(UserDefinedTypeName const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(FunctionTypeName const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Mapping const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ArrayTypeName const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(InlineAssembly const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Block const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(PlaceholderStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(IfStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(WhileStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ForStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Continue const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Break const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Return const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Throw const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(EmitStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(VariableDeclarationStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ExpressionStatement const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Conditional const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Assignment const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(TupleExpression const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(UnaryOperation const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(BinaryOperation const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(FunctionCall const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(NewExpression const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(MemberAccess const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(IndexAccess const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Identifier const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(ElementaryTypeNameExpression const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::endVisit(Literal const&)
 | |
| {
 | |
| 	m_indentation--;
 | |
| }
 | |
| 
 | |
| void ASTPrinter::printSourcePart(ASTNode const& _node)
 | |
| {
 | |
| 	if (m_gasCosts.count(&_node))
 | |
| 		*m_ostream << indentation() << "   Gas costs: " << m_gasCosts.at(&_node) << endl;
 | |
| 	if (!m_source.empty())
 | |
| 	{
 | |
| 		SourceLocation const& location(_node.location());
 | |
| 		*m_ostream <<
 | |
| 			indentation() <<
 | |
| 			"   Source: " <<
 | |
| 			Json::valueToQuotedString(m_source.substr(location.start, location.end - location.start).c_str()) <<
 | |
| 			endl;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| void ASTPrinter::printType(Expression const& _expression)
 | |
| {
 | |
| 	if (_expression.annotation().type)
 | |
| 		*m_ostream << indentation() << "   Type: " << _expression.annotation().type->toString() << "\n";
 | |
| 	else
 | |
| 		*m_ostream << indentation() << "   Type unknown.\n";
 | |
| }
 | |
| 
 | |
| string ASTPrinter::indentation() const
 | |
| {
 | |
| 	return string(m_indentation * 2, ' ');
 | |
| }
 | |
| 
 | |
| void ASTPrinter::writeLine(string const& _line)
 | |
| {
 | |
| 	*m_ostream << indentation() << _line << endl;
 | |
| }
 | |
| 
 | |
| }
 | |
| }
 |