xref: /llvm-project/flang/examples/PrintFlangFunctionNames/PrintFlangFunctionNames.cpp (revision ea18987094eff5a5835135da1f472d2e7bf6c68e)
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