1520e5db2SStuart Ellis //===-- PrintFlangFunctionNames.cpp ---------------------------------------===// 2520e5db2SStuart Ellis // 3520e5db2SStuart Ellis // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4520e5db2SStuart Ellis // See https://llvm.org/LICENSE.txt for license information. 5520e5db2SStuart Ellis // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6520e5db2SStuart Ellis // 7520e5db2SStuart Ellis //===----------------------------------------------------------------------===// 8520e5db2SStuart Ellis // 9520e5db2SStuart Ellis // Small example Flang plugin to count/print Functions & Subroutines names. 10520e5db2SStuart Ellis // It walks the Parse Tree using a Visitor struct that has Post functions for 11520e5db2SStuart Ellis // FunctionStmt and SubroutineStmt to access the names of functions & 12520e5db2SStuart Ellis // subroutines. It also has Pre functions for FunctionSubprogram and 13520e5db2SStuart Ellis // SubroutineSubprogram so a Bool can be set to show that it is the definition 14520e5db2SStuart Ellis // of a function/subroutine, and not print those that are in an Interface. 15520e5db2SStuart Ellis // This plugin does not recognise Statement Functions or Module Procedures, 16520e5db2SStuart Ellis // which could be dealt with through StmtFunctionStmt and MpSubprogramStmt nodes 17520e5db2SStuart Ellis // respectively. 18520e5db2SStuart Ellis // 19520e5db2SStuart Ellis //===----------------------------------------------------------------------===// 20520e5db2SStuart Ellis 21520e5db2SStuart Ellis #include "flang/Frontend/FrontendActions.h" 22520e5db2SStuart Ellis #include "flang/Frontend/FrontendPluginRegistry.h" 23520e5db2SStuart Ellis #include "flang/Parser/dump-parse-tree.h" 24520e5db2SStuart Ellis #include "flang/Parser/parsing.h" 25520e5db2SStuart Ellis 26520e5db2SStuart Ellis using namespace Fortran::frontend; 27520e5db2SStuart Ellis 28520e5db2SStuart Ellis class PrintFunctionNamesAction : public PluginParseTreeAction { 29520e5db2SStuart Ellis 30520e5db2SStuart Ellis // Visitor struct that defines Pre/Post functions for different types of nodes 31520e5db2SStuart Ellis struct ParseTreeVisitor { PrePrintFunctionNamesAction::ParseTreeVisitor32520e5db2SStuart Ellis template <typename A> bool Pre(const A &) { return true; } PostPrintFunctionNamesAction::ParseTreeVisitor33520e5db2SStuart Ellis template <typename A> void Post(const A &) {} 34520e5db2SStuart Ellis PrePrintFunctionNamesAction::ParseTreeVisitor35520e5db2SStuart Ellis bool Pre(const Fortran::parser::FunctionSubprogram &) { 36520e5db2SStuart Ellis isInSubprogram_ = true; 37520e5db2SStuart Ellis return true; 38520e5db2SStuart Ellis } PostPrintFunctionNamesAction::ParseTreeVisitor39520e5db2SStuart Ellis void Post(const Fortran::parser::FunctionStmt &f) { 40520e5db2SStuart Ellis if (isInSubprogram_) { 41520e5db2SStuart Ellis llvm::outs() << "Function:\t" 42520e5db2SStuart Ellis << std::get<Fortran::parser::Name>(f.t).ToString() << "\n"; 43520e5db2SStuart Ellis fcounter++; 44520e5db2SStuart Ellis isInSubprogram_ = false; 45520e5db2SStuart Ellis } 46520e5db2SStuart Ellis } 47520e5db2SStuart Ellis PrePrintFunctionNamesAction::ParseTreeVisitor48520e5db2SStuart Ellis bool Pre(const Fortran::parser::SubroutineSubprogram &) { 49520e5db2SStuart Ellis isInSubprogram_ = true; 50520e5db2SStuart Ellis return true; 51520e5db2SStuart Ellis } PostPrintFunctionNamesAction::ParseTreeVisitor52520e5db2SStuart Ellis void Post(const Fortran::parser::SubroutineStmt &s) { 53520e5db2SStuart Ellis if (isInSubprogram_) { 54520e5db2SStuart Ellis llvm::outs() << "Subroutine:\t" 55520e5db2SStuart Ellis << std::get<Fortran::parser::Name>(s.t).ToString() << "\n"; 56520e5db2SStuart Ellis scounter++; 57520e5db2SStuart Ellis isInSubprogram_ = false; 58520e5db2SStuart Ellis } 59520e5db2SStuart Ellis } 60520e5db2SStuart Ellis 61520e5db2SStuart Ellis int fcounter{0}; 62520e5db2SStuart Ellis int scounter{0}; 63520e5db2SStuart Ellis 64520e5db2SStuart Ellis private: 65520e5db2SStuart Ellis bool isInSubprogram_{false}; 66520e5db2SStuart Ellis }; 67520e5db2SStuart Ellis executeAction()68*ea189870SAndrzej Warzynski void executeAction() override { 69520e5db2SStuart Ellis ParseTreeVisitor visitor; 702186a4aeSAndrzej Warzynski Fortran::parser::Walk(getParsing().parseTree(), visitor); 71520e5db2SStuart Ellis 72520e5db2SStuart Ellis llvm::outs() << "\n==== Functions: " << visitor.fcounter << " ====\n"; 73520e5db2SStuart Ellis llvm::outs() << "==== Subroutines: " << visitor.scounter << " ====\n"; 74520e5db2SStuart Ellis } 75520e5db2SStuart Ellis }; 76520e5db2SStuart Ellis 77520e5db2SStuart Ellis static FrontendPluginRegistry::Add<PrintFunctionNamesAction> X( 78520e5db2SStuart Ellis "print-fns", "Print Function names"); 79