[TargetVersion] Only enable on RISC-V and AArch64 (#115991)
[llvm-project.git] / clang / docs / LibASTMatchersTutorial.rst
blobd2883688ebfacc5bce16c1b0fe4dbb8646f9bb5e
1 ===============================================================
2 Tutorial for building tools using LibTooling and LibASTMatchers
3 ===============================================================
5 This document is intended to show how to build a useful source-to-source
6 translation tool based on Clang's `LibTooling <LibTooling.html>`_. It is
7 explicitly aimed at people who are new to Clang, so all you should need
8 is a working knowledge of C++ and the command line.
10 In order to work on the compiler, you need some basic knowledge of the
11 abstract syntax tree (AST). To this end, the reader is encouraged to
12 skim the :doc:`Introduction to the Clang
13 AST <IntroductionToTheClangAST>`
15 Step 0: Obtaining Clang
16 =======================
18 As Clang is part of the LLVM project, you'll need to download LLVM's
19 source code first. Both Clang and LLVM are in the same git repository,
20 under different directories. For further information, see the `getting
21 started guide <https://llvm.org/docs/GettingStarted.html>`_.
23 .. code-block:: console
25       mkdir ~/clang-llvm && cd ~/clang-llvm
26       git clone https://github.com/llvm/llvm-project.git
28 Next you need to obtain the CMake build system and Ninja build tool.
30 .. code-block:: console
32       cd ~/clang-llvm
33       git clone https://github.com/martine/ninja.git
34       cd ninja
35       git checkout release
36       ./configure.py --bootstrap
37       sudo cp ninja /usr/bin/
39       cd ~/clang-llvm
40       git clone https://gitlab.kitware.com/cmake/cmake.git
41       cd cmake
42       git checkout next
43       ./bootstrap
44       make
45       sudo make install
47 Okay. Now we'll build Clang!
49 .. code-block:: console
51       cd ~/clang-llvm
52       mkdir build && cd build
53       cmake -G Ninja ../llvm-project/llvm -DLLVM_ENABLE_PROJECTS="clang;clang-tools-extra" -DCMAKE_BUILD_TYPE=Release -DLLVM_BUILD_TESTS=ON
54       ninja
55       ninja check       # Test LLVM only.
56       ninja clang-test  # Test Clang only.
57       ninja install
59 And we're live.
61 All of the tests should pass.
63 Finally, we want to set Clang as its own compiler.
65 .. code-block:: console
67       cd ~/clang-llvm/build
68       ccmake ../llvm-project/llvm
70 The second command will bring up a GUI for configuring Clang. You need
71 to set the entry for ``CMAKE_CXX_COMPILER``. Press ``'t'`` to turn on
72 advanced mode. Scroll down to ``CMAKE_CXX_COMPILER``, and set it to
73 ``/usr/bin/clang++``, or wherever you installed it. Press ``'c'`` to
74 configure, then ``'g'`` to generate CMake's files.
76 Finally, run ninja one last time, and you're done.
78 Step 1: Create a ClangTool
79 ==========================
81 Now that we have enough background knowledge, it's time to create the
82 simplest productive ClangTool in existence: a syntax checker. While this
83 already exists as ``clang-check``, it's important to understand what's
84 going on.
86 First, we'll need to create a new directory for our tool and tell CMake
87 that it exists. As this is not going to be a core clang tool, it will
88 live in the ``clang-tools-extra`` repository.
90 .. code-block:: console
92       cd ~/clang-llvm/llvm-project
93       mkdir clang-tools-extra/loop-convert
94       echo 'add_subdirectory(loop-convert)' >> clang-tools-extra/CMakeLists.txt
95       vim clang-tools-extra/loop-convert/CMakeLists.txt
97 CMakeLists.txt should have the following contents:
101       set(LLVM_LINK_COMPONENTS support)
103       add_clang_executable(loop-convert
104         LoopConvert.cpp
105         )
106       target_link_libraries(loop-convert
107         PRIVATE
108         clangAST
109         clangASTMatchers
110         clangBasic
111         clangFrontend
112         clangSerialization
113         clangTooling
114         )
116 With that done, Ninja will be able to compile our tool. Let's give it
117 something to compile! Put the following into
118 ``clang-tools-extra/loop-convert/LoopConvert.cpp``. A detailed explanation of
119 why the different parts are needed can be found in the `LibTooling
120 documentation <LibTooling.html>`_.
122 .. code-block:: c++
124       // Declares clang::SyntaxOnlyAction.
125       #include "clang/Frontend/FrontendActions.h"
126       #include "clang/Tooling/CommonOptionsParser.h"
127       #include "clang/Tooling/Tooling.h"
128       // Declares llvm::cl::extrahelp.
129       #include "llvm/Support/CommandLine.h"
131       using namespace clang::tooling;
132       using namespace llvm;
134       // Apply a custom category to all command-line options so that they are the
135       // only ones displayed.
136       static llvm::cl::OptionCategory MyToolCategory("my-tool options");
138       // CommonOptionsParser declares HelpMessage with a description of the common
139       // command-line options related to the compilation database and input files.
140       // It's nice to have this help message in all tools.
141       static cl::extrahelp CommonHelp(CommonOptionsParser::HelpMessage);
143       // A help message for this specific tool can be added afterwards.
144       static cl::extrahelp MoreHelp("\nMore help text...\n");
146       int main(int argc, const char **argv) {
147         auto ExpectedParser = CommonOptionsParser::create(argc, argv, MyToolCategory);
148         if (!ExpectedParser) {
149           // Fail gracefully for unsupported options.
150           llvm::errs() << ExpectedParser.takeError();
151           return 1;
152         }
153         CommonOptionsParser& OptionsParser = ExpectedParser.get();
154         ClangTool Tool(OptionsParser.getCompilations(),
155                        OptionsParser.getSourcePathList());
156         return Tool.run(newFrontendActionFactory<clang::SyntaxOnlyAction>().get());
157       }
159 And that's it! You can compile our new tool by running ninja from the
160 ``build`` directory.
162 .. code-block:: console
164       cd ~/clang-llvm/build
165       ninja
167 You should now be able to run the syntax checker, which is located in
168 ``~/clang-llvm/build/bin``, on any source file. Try it!
170 .. code-block:: console
172       echo "int main() { return 0; }" > test.cpp
173       bin/loop-convert test.cpp --
175 Note the two dashes after we specify the source file. The additional
176 options for the compiler are passed after the dashes rather than loading
177 them from a compilation database - there just aren't any options needed
178 right now.
180 Intermezzo: Learn AST matcher basics
181 ====================================
183 Clang recently introduced the :doc:`ASTMatcher
184 library <LibASTMatchers>` to provide a simple, powerful, and
185 concise way to describe specific patterns in the AST. Implemented as a
186 DSL powered by macros and templates (see
187 `ASTMatchers.h <../doxygen/ASTMatchers_8h_source.html>`_ if you're
188 curious), matchers offer the feel of algebraic data types common to
189 functional programming languages.
191 For example, suppose you wanted to examine only binary operators. There
192 is a matcher to do exactly that, conveniently named ``binaryOperator``.
193 I'll give you one guess what this matcher does:
195 .. code-block:: c++
197       binaryOperator(hasOperatorName("+"), hasLHS(integerLiteral(equals(0))))
199 Shockingly, it will match against addition expressions whose left hand
200 side is exactly the literal 0. It will not match against other forms of
201 0, such as ``'\0'`` or ``NULL``, but it will match against macros that
202 expand to 0. The matcher will also not match against calls to the
203 overloaded operator ``'+'``, as there is a separate ``operatorCallExpr``
204 matcher to handle overloaded operators.
206 There are AST matchers to match all the different nodes of the AST,
207 narrowing matchers to only match AST nodes fulfilling specific criteria,
208 and traversal matchers to get from one kind of AST node to another. For
209 a complete list of AST matchers, take a look at the `AST Matcher
210 References <LibASTMatchersReference.html>`_
212 All matcher that are nouns describe entities in the AST and can be
213 bound, so that they can be referred to whenever a match is found. To do
214 so, simply call the method ``bind`` on these matchers, e.g.:
216 .. code-block:: c++
218       variable(hasType(isInteger())).bind("intvar")
220 Step 2: Using AST matchers
221 ==========================
223 Okay, on to using matchers for real. Let's start by defining a matcher
224 which will capture all ``for`` statements that define a new variable
225 initialized to zero. Let's start with matching all ``for`` loops:
227 .. code-block:: c++
229       forStmt()
231 Next, we want to specify that a single variable is declared in the first
232 portion of the loop, so we can extend the matcher to
234 .. code-block:: c++
236       forStmt(hasLoopInit(declStmt(hasSingleDecl(varDecl()))))
238 Finally, we can add the condition that the variable is initialized to
239 zero.
241 .. code-block:: c++
243       forStmt(hasLoopInit(declStmt(hasSingleDecl(varDecl(
244         hasInitializer(integerLiteral(equals(0))))))))
246 It is fairly easy to read and understand the matcher definition ("match
247 loops whose init portion declares a single variable which is initialized
248 to the integer literal 0"), but deciding that every piece is necessary
249 is more difficult. Note that this matcher will not match loops whose
250 variables are initialized to ``'\0'``, ``0.0``, ``NULL``, or any form of
251 zero besides the integer 0.
253 The last step is giving the matcher a name and binding the ``ForStmt``
254 as we will want to do something with it:
256 .. code-block:: c++
258       StatementMatcher LoopMatcher =
259         forStmt(hasLoopInit(declStmt(hasSingleDecl(varDecl(
260           hasInitializer(integerLiteral(equals(0)))))))).bind("forLoop");
262 Once you have defined your matchers, you will need to add a little more
263 scaffolding in order to run them. Matchers are paired with a
264 ``MatchCallback`` and registered with a ``MatchFinder`` object, then run
265 from a ``ClangTool``. More code!
267 Add the following to ``LoopConvert.cpp``:
269 .. code-block:: c++
271       #include "clang/ASTMatchers/ASTMatchers.h"
272       #include "clang/ASTMatchers/ASTMatchFinder.h"
274       using namespace clang;
275       using namespace clang::ast_matchers;
277       StatementMatcher LoopMatcher =
278         forStmt(hasLoopInit(declStmt(hasSingleDecl(varDecl(
279           hasInitializer(integerLiteral(equals(0)))))))).bind("forLoop");
281       class LoopPrinter : public MatchFinder::MatchCallback {
282       public :
283         virtual void run(const MatchFinder::MatchResult &Result) {
284           if (const ForStmt *FS = Result.Nodes.getNodeAs<clang::ForStmt>("forLoop"))
285             FS->dump();
286         }
287       };
289 And change ``main()`` to:
291 .. code-block:: c++
293       int main(int argc, const char **argv) {
294         auto ExpectedParser = CommonOptionsParser::create(argc, argv, MyToolCategory);
295         if (!ExpectedParser) {
296           // Fail gracefully for unsupported options.
297           llvm::errs() << ExpectedParser.takeError();
298           return 1;
299         }
300         CommonOptionsParser& OptionsParser = ExpectedParser.get();
301         ClangTool Tool(OptionsParser.getCompilations(),
302                        OptionsParser.getSourcePathList());
304         LoopPrinter Printer;
305         MatchFinder Finder;
306         Finder.addMatcher(LoopMatcher, &Printer);
308         return Tool.run(newFrontendActionFactory(&Finder).get());
309       }
311 Now, you should be able to recompile and run the code to discover for
312 loops. Create a new file with a few examples, and test out our new
313 handiwork:
315 .. code-block:: console
317       cd ~/clang-llvm/build/
318       ninja loop-convert
319       vim ~/test-files/simple-loops.cc
320       bin/loop-convert ~/test-files/simple-loops.cc
322 Step 3.5: More Complicated Matchers
323 ===================================
325 Our simple matcher is capable of discovering for loops, but we would
326 still need to filter out many more ourselves. We can do a good portion
327 of the remaining work with some cleverly chosen matchers, but first we
328 need to decide exactly which properties we want to allow.
330 How can we characterize for loops over arrays which would be eligible
331 for translation to range-based syntax? Range based loops over arrays of
332 size ``N`` that:
334 -  start at index ``0``
335 -  iterate consecutively
336 -  end at index ``N-1``
338 We already check for (1), so all we need to add is a check to the loop's
339 condition to ensure that the loop's index variable is compared against
340 ``N`` and another check to ensure that the increment step just
341 increments this same variable. The matcher for (2) is straightforward:
342 require a pre- or post-increment of the same variable declared in the
343 init portion.
345 Unfortunately, such a matcher is impossible to write. Matchers contain
346 no logic for comparing two arbitrary AST nodes and determining whether
347 or not they are equal, so the best we can do is matching more than we
348 would like to allow, and punting extra comparisons to the callback.
350 In any case, we can start building this sub-matcher. We can require that
351 the increment step be a unary increment like this:
353 .. code-block:: c++
355       hasIncrement(unaryOperator(hasOperatorName("++")))
357 Specifying what is incremented introduces another quirk of Clang's AST:
358 Usages of variables are represented as ``DeclRefExpr``'s ("declaration
359 reference expressions") because they are expressions which refer to
360 variable declarations. To find a ``unaryOperator`` that refers to a
361 specific declaration, we can simply add a second condition to it:
363 .. code-block:: c++
365       hasIncrement(unaryOperator(
366         hasOperatorName("++"),
367         hasUnaryOperand(declRefExpr())))
369 Furthermore, we can restrict our matcher to only match if the
370 incremented variable is an integer:
372 .. code-block:: c++
374       hasIncrement(unaryOperator(
375         hasOperatorName("++"),
376         hasUnaryOperand(declRefExpr(to(varDecl(hasType(isInteger())))))))
378 And the last step will be to attach an identifier to this variable, so
379 that we can retrieve it in the callback:
381 .. code-block:: c++
383       hasIncrement(unaryOperator(
384         hasOperatorName("++"),
385         hasUnaryOperand(declRefExpr(to(
386           varDecl(hasType(isInteger())).bind("incrementVariable"))))))
388 We can add this code to the definition of ``LoopMatcher`` and make sure
389 that our program, outfitted with the new matcher, only prints out loops
390 that declare a single variable initialized to zero and have an increment
391 step consisting of a unary increment of some variable.
393 Now, we just need to add a matcher to check if the condition part of the
394 ``for`` loop compares a variable against the size of the array. There is
395 only one problem - we don't know which array we're iterating over
396 without looking at the body of the loop! We are again restricted to
397 approximating the result we want with matchers, filling in the details
398 in the callback. So we start with:
400 .. code-block:: c++
402       hasCondition(binaryOperator(hasOperatorName("<")))
404 It makes sense to ensure that the left-hand side is a reference to a
405 variable, and that the right-hand side has integer type.
407 .. code-block:: c++
409       hasCondition(binaryOperator(
410         hasOperatorName("<"),
411         hasLHS(declRefExpr(to(varDecl(hasType(isInteger()))))),
412         hasRHS(expr(hasType(isInteger())))))
414 Why? Because it doesn't work. Of the three loops provided in
415 ``test-files/simple.cpp``, zero of them have a matching condition. A
416 quick look at the AST dump of the first for loop, produced by the
417 previous iteration of loop-convert, shows us the answer:
421       (ForStmt 0x173b240
422         (DeclStmt 0x173afc8
423           0x173af50 "int i =
424             (IntegerLiteral 0x173afa8 'int' 0)")
425         <<>>
426         (BinaryOperator 0x173b060 '_Bool' '<'
427           (ImplicitCastExpr 0x173b030 'int'
428             (DeclRefExpr 0x173afe0 'int' lvalue Var 0x173af50 'i' 'int'))
429           (ImplicitCastExpr 0x173b048 'int'
430             (DeclRefExpr 0x173b008 'const int' lvalue Var 0x170fa80 'N' 'const int')))
431         (UnaryOperator 0x173b0b0 'int' lvalue prefix '++'
432           (DeclRefExpr 0x173b088 'int' lvalue Var 0x173af50 'i' 'int'))
433         (CompoundStatement ...
435 We already know that the declaration and increments both match, or this
436 loop wouldn't have been dumped. The culprit lies in the implicit cast
437 applied to the first operand (i.e. the LHS) of the less-than operator,
438 an L-value to R-value conversion applied to the expression referencing
439 ``i``. Thankfully, the matcher library offers a solution to this problem
440 in the form of ``ignoringParenImpCasts``, which instructs the matcher to
441 ignore implicit casts and parentheses before continuing to match.
442 Adjusting the condition operator will restore the desired match.
444 .. code-block:: c++
446       hasCondition(binaryOperator(
447         hasOperatorName("<"),
448         hasLHS(ignoringParenImpCasts(declRefExpr(
449           to(varDecl(hasType(isInteger())))))),
450         hasRHS(expr(hasType(isInteger())))))
452 After adding binds to the expressions we wished to capture and
453 extracting the identifier strings into variables, we have array-step-2
454 completed.
456 Step 4: Retrieving Matched Nodes
457 ================================
459 So far, the matcher callback isn't very interesting: it just dumps the
460 loop's AST. At some point, we will need to make changes to the input
461 source code. Next, we'll work on using the nodes we bound in the
462 previous step.
464 The ``MatchFinder::run()`` callback takes a
465 ``MatchFinder::MatchResult&`` as its parameter. We're most interested in
466 its ``Context`` and ``Nodes`` members. Clang uses the ``ASTContext``
467 class to represent contextual information about the AST, as the name
468 implies, though the most functionally important detail is that several
469 operations require an ``ASTContext*`` parameter. More immediately useful
470 is the set of matched nodes, and how we retrieve them.
472 Since we bind three variables (identified by ConditionVarName,
473 InitVarName, and IncrementVarName), we can obtain the matched nodes by
474 using the ``getNodeAs()`` member function.
476 In ``LoopConvert.cpp`` add
478 .. code-block:: c++
480       #include "clang/AST/ASTContext.h"
482 Change ``LoopMatcher`` to
484 .. code-block:: c++
486       StatementMatcher LoopMatcher =
487           forStmt(hasLoopInit(declStmt(
488                       hasSingleDecl(varDecl(hasInitializer(integerLiteral(equals(0))))
489                                         .bind("initVarName")))),
490                   hasIncrement(unaryOperator(
491                       hasOperatorName("++"),
492                       hasUnaryOperand(declRefExpr(
493                           to(varDecl(hasType(isInteger())).bind("incVarName")))))),
494                   hasCondition(binaryOperator(
495                       hasOperatorName("<"),
496                       hasLHS(ignoringParenImpCasts(declRefExpr(
497                           to(varDecl(hasType(isInteger())).bind("condVarName"))))),
498                       hasRHS(expr(hasType(isInteger())))))).bind("forLoop");
500 And change ``LoopPrinter::run`` to
502 .. code-block:: c++
504       void LoopPrinter::run(const MatchFinder::MatchResult &Result) {
505         ASTContext *Context = Result.Context;
506         const ForStmt *FS = Result.Nodes.getNodeAs<ForStmt>("forLoop");
507         // We do not want to convert header files!
508         if (!FS || !Context->getSourceManager().isWrittenInMainFile(FS->getForLoc()))
509           return;
510         const VarDecl *IncVar = Result.Nodes.getNodeAs<VarDecl>("incVarName");
511         const VarDecl *CondVar = Result.Nodes.getNodeAs<VarDecl>("condVarName");
512         const VarDecl *InitVar = Result.Nodes.getNodeAs<VarDecl>("initVarName");
514         if (!areSameVariable(IncVar, CondVar) || !areSameVariable(IncVar, InitVar))
515           return;
516         llvm::outs() << "Potential array-based loop discovered.\n";
517       }
519 Clang associates a ``VarDecl`` with each variable to represent the variable's
520 declaration. Since the "canonical" form of each declaration is unique by
521 address, all we need to do is make sure neither ``ValueDecl`` (base class of
522 ``VarDecl``) is ``NULL`` and compare the canonical Decls.
524 .. code-block:: c++
526       static bool areSameVariable(const ValueDecl *First, const ValueDecl *Second) {
527         return First && Second &&
528                First->getCanonicalDecl() == Second->getCanonicalDecl();
529       }
531 If execution reaches the end of ``LoopPrinter::run()``, we know that the
532 loop shell looks like
534 .. code-block:: c++
536       for (int i= 0; i < expr(); ++i) { ... }
538 For now, we will just print a message explaining that we found a loop.
539 The next section will deal with recursively traversing the AST to
540 discover all changes needed.
542 As a side note, it's not as trivial to test if two expressions are the same,
543 though Clang has already done the hard work for us by providing a way to
544 canonicalize expressions:
546 .. code-block:: c++
548       static bool areSameExpr(ASTContext *Context, const Expr *First,
549                               const Expr *Second) {
550         if (!First || !Second)
551           return false;
552         llvm::FoldingSetNodeID FirstID, SecondID;
553         First->Profile(FirstID, *Context, true);
554         Second->Profile(SecondID, *Context, true);
555         return FirstID == SecondID;
556       }
558 This code relies on the comparison between two
559 ``llvm::FoldingSetNodeIDs``. As the documentation for
560 ``Stmt::Profile()`` indicates, the ``Profile()`` member function builds
561 a description of a node in the AST, based on its properties, along with
562 those of its children. ``FoldingSetNodeID`` then serves as a hash we can
563 use to compare expressions. We will need ``areSameExpr`` later. Before
564 you run the new code on the additional loops added to
565 test-files/simple.cpp, try to figure out which ones will be considered
566 potentially convertible.