|  | // Copyright (c) 2020 Google LLC | 
|  | // | 
|  | // Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | // you may not use this file except in compliance with the License. | 
|  | // You may obtain a copy of the License at | 
|  | // | 
|  | //     http://www.apache.org/licenses/LICENSE-2.0 | 
|  | // | 
|  | // Unless required by applicable law or agreed to in writing, software | 
|  | // distributed under the License is distributed on an "AS IS" BASIS, | 
|  | // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | // See the License for the specific language governing permissions and | 
|  | // limitations under the License. | 
|  |  | 
|  | #include "source/fuzz/call_graph.h" | 
|  |  | 
|  | #include <queue> | 
|  |  | 
|  | namespace spvtools { | 
|  | namespace fuzz { | 
|  |  | 
|  | CallGraph::CallGraph(opt::IRContext* context) { | 
|  | // Initialize function in-degree and call graph edges to 0 and empty. | 
|  | for (auto& function : *context->module()) { | 
|  | function_in_degree_[function.result_id()] = 0; | 
|  | call_graph_edges_[function.result_id()] = std::set<uint32_t>(); | 
|  | } | 
|  |  | 
|  | // Consider every function. | 
|  | for (auto& function : *context->module()) { | 
|  | // Avoid considering the same callee of this function multiple times by | 
|  | // recording known callees. | 
|  | std::set<uint32_t> known_callees; | 
|  | // Consider every function call instruction in every block. | 
|  | for (auto& block : function) { | 
|  | for (auto& instruction : block) { | 
|  | if (instruction.opcode() != SpvOpFunctionCall) { | 
|  | continue; | 
|  | } | 
|  | // Get the id of the function being called. | 
|  | uint32_t callee = instruction.GetSingleWordInOperand(0); | 
|  | if (known_callees.count(callee)) { | 
|  | // We have already considered a call to this function - ignore it. | 
|  | continue; | 
|  | } | 
|  | // Increase the callee's in-degree and add an edge to the call graph. | 
|  | function_in_degree_[callee]++; | 
|  | call_graph_edges_[function.result_id()].insert(callee); | 
|  | // Mark the callee as 'known'. | 
|  | known_callees.insert(callee); | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void CallGraph::PushDirectCallees(uint32_t function_id, | 
|  | std::queue<uint32_t>* queue) const { | 
|  | for (auto callee : GetDirectCallees(function_id)) { | 
|  | queue->push(callee); | 
|  | } | 
|  | } | 
|  |  | 
|  | std::set<uint32_t> CallGraph::GetIndirectCallees(uint32_t function_id) const { | 
|  | std::set<uint32_t> result; | 
|  | std::queue<uint32_t> queue; | 
|  | PushDirectCallees(function_id, &queue); | 
|  |  | 
|  | while (!queue.empty()) { | 
|  | auto next = queue.front(); | 
|  | queue.pop(); | 
|  | if (result.count(next)) { | 
|  | continue; | 
|  | } | 
|  | result.insert(next); | 
|  | PushDirectCallees(next, &queue); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | }  // namespace fuzz | 
|  | }  // namespace spvtools |