mirror of
				https://github.com/ethereum/solidity
				synced 2023-10-03 13:03:40 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			95 lines
		
	
	
		
			3.8 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			95 lines
		
	
	
		
			3.8 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/>.
 | |
| */
 | |
| // SPDX-License-Identifier: GPL-3.0
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <libsolidity/ast/ASTForward.h>
 | |
| #include <libsolidity/ast/ASTVisitor.h>
 | |
| #include <libsolidity/ast/CallGraph.h>
 | |
| 
 | |
| #include <deque>
 | |
| #include <ostream>
 | |
| 
 | |
| namespace solidity::frontend
 | |
| {
 | |
| 
 | |
| /**
 | |
|  * Creates a function call graph for a contract at the granularity of Solidity functions and modifiers.
 | |
|  * or after deployment. The graph does not preserve temporal relations between calls - edges
 | |
|  * coming out of the same node show which calls were performed but not in what order.
 | |
|  *
 | |
|  * Includes the following special nodes:
 | |
|  *  - Entry: represents a call from the outside of the contract.
 | |
|  *    After deployment this is the node that connects to all the functions exposed through the
 | |
|  *    external interface. At contract creation it connects to the constructors and variable
 | |
|  *    initializers, which are not explicitly called from within another function.
 | |
|  *  - InternalDispatch: Represents the internal dispatch function, which calls internal functions
 | |
|  *    determined at runtime by values of variables and expressions. Functions that are not called
 | |
|  *    right away get an edge from this node.
 | |
|  *
 | |
|  *  Nodes are a variant of either the enum SpecialNode or a CallableDeclaration which currently
 | |
|  *  can be a function or a modifier. There are no nodes representing event calls. Instead all
 | |
|  *  emitted events and created contracts are gathered in separate sets included in the graph just
 | |
|  *  for that purpose.
 | |
|  *
 | |
|  *  Auto-generated getter functions for public state variables are ignored, but function calls
 | |
|  *  inside initial assignments are included in the creation graph.
 | |
|  *
 | |
|  *  Only calls reachable from an Entry node are included in the graph. The map representing edges
 | |
|  *  is also guaranteed to contain keys representing all the reachable functions and modifiers, even
 | |
|  *  if they have no outgoing edges.
 | |
|  */
 | |
| class FunctionCallGraphBuilder: private ASTConstVisitor
 | |
| {
 | |
| public:
 | |
| 	static CallGraph buildCreationGraph(ContractDefinition const& _contract);
 | |
| 	static CallGraph buildDeployedGraph(
 | |
| 		ContractDefinition const& _contract,
 | |
| 		CallGraph const& _creationGraph
 | |
| 	);
 | |
| 
 | |
| private:
 | |
| 	FunctionCallGraphBuilder(ContractDefinition const& _contract):
 | |
| 		m_contract(_contract)
 | |
| 	{}
 | |
| 
 | |
| 	bool visit(FunctionCall const& _functionCall) override;
 | |
| 	bool visit(EmitStatement const& _emitStatement) override;
 | |
| 	bool visit(Identifier const& _identifier) override;
 | |
| 	bool visit(MemberAccess const& _memberAccess) override;
 | |
| 	bool visit(BinaryOperation const& _binaryOperation) override;
 | |
| 	bool visit(UnaryOperation const& _unaryOperation) override;
 | |
| 	bool visit(ModifierInvocation const& _modifierInvocation) override;
 | |
| 	bool visit(NewExpression const& _newExpression) override;
 | |
| 
 | |
| 	void enqueueCallable(CallableDeclaration const& _callable);
 | |
| 	void processQueue();
 | |
| 
 | |
| 	void add(CallGraph::Node _caller, CallGraph::Node _callee);
 | |
| 	void functionReferenced(CallableDeclaration const& _callable, bool _calledDirectly = true);
 | |
| 
 | |
| 	CallGraph::Node m_currentNode = CallGraph::SpecialNode::Entry;
 | |
| 	ContractDefinition const& m_contract;
 | |
| 	CallGraph m_graph;
 | |
| 	std::deque<CallableDeclaration const*> m_visitQueue;
 | |
| };
 | |
| 
 | |
| std::ostream& operator<<(std::ostream& _out, CallGraph::Node const& _node);
 | |
| 
 | |
| }
 |