forked from OSchip/llvm-project
[docs] Make WritingAnLLVMPass.rst up-to-date with current state of things
This patch updates WritingAnLLVMPass.rst to make it in line with current state of things. Specifically: * Makefile instructions replaced with CMake ones * Filenames replaced with correct ones * Example reformatted a bit to make it less confusing and more conforming to LLVM Coding Standards * opt tool output updated with what it actually prints nowdays * "gcse" (which doesn't exist anymore) replaced with "gvn" (which still does) Differential Revision: https://reviews.llvm.org/D24233 llvm-svn: 282482
This commit is contained in:
parent
10e2b5dcaa
commit
3d3ae6f496
|
@ -50,34 +50,34 @@ Setting up the build environment
|
||||||
First, configure and build LLVM. Next, you need to create a new directory
|
First, configure and build LLVM. Next, you need to create a new directory
|
||||||
somewhere in the LLVM source base. For this example, we'll assume that you
|
somewhere in the LLVM source base. For this example, we'll assume that you
|
||||||
made ``lib/Transforms/Hello``. Finally, you must set up a build script
|
made ``lib/Transforms/Hello``. Finally, you must set up a build script
|
||||||
(``Makefile``) that will compile the source code for the new pass. To do this,
|
that will compile the source code for the new pass. To do this,
|
||||||
copy the following into ``Makefile``:
|
copy the following into ``CMakeLists.txt``:
|
||||||
|
|
||||||
.. code-block:: make
|
.. code-block:: cmake
|
||||||
|
|
||||||
# Makefile for hello pass
|
add_llvm_loadable_module( LLVMHello
|
||||||
|
Hello.cpp
|
||||||
|
|
||||||
|
PLUGIN_TOOL
|
||||||
|
opt
|
||||||
|
)
|
||||||
|
|
||||||
# Path to top level of LLVM hierarchy
|
and the following line into ``lib/Transforms/CMakeLists.txt``:
|
||||||
LEVEL = ../../..
|
|
||||||
|
|
||||||
# Name of the library to build
|
.. code-block:: cmake
|
||||||
LIBRARYNAME = Hello
|
|
||||||
|
|
||||||
# Make the shared library become a loadable module so the tools can
|
add_subdirectory(Hello)
|
||||||
# dlopen/dlsym on the resulting library.
|
|
||||||
LOADABLE_MODULE = 1
|
|
||||||
|
|
||||||
# Include the makefile implementation stuff
|
(Note that there is already a directory named ``Hello`` with a sample "Hello"
|
||||||
include $(LEVEL)/Makefile.common
|
pass; you may play with it -- in which case you don't need to modify any
|
||||||
|
``CMakeLists.txt`` files -- or, if you want to create everything from scratch,
|
||||||
|
use another name.)
|
||||||
|
|
||||||
This makefile specifies that all of the ``.cpp`` files in the current directory
|
This build script specifies that ``Hello.cpp`` file in the current directory
|
||||||
are to be compiled and linked together into a shared object
|
is to be compiled and linked into a shared object ``$(LEVEL)/lib/LLVMHello.so`` that
|
||||||
``$(LEVEL)/Debug+Asserts/lib/Hello.so`` that can be dynamically loaded by the
|
can be dynamically loaded by the :program:`opt` tool via its :option:`-load`
|
||||||
:program:`opt` or :program:`bugpoint` tools via their :option:`-load` options.
|
option. If your operating system uses a suffix other than ``.so`` (such as
|
||||||
If your operating system uses a suffix other than ``.so`` (such as Windows or Mac
|
Windows or Mac OS X), the appropriate extension will be used.
|
||||||
OS X), the appropriate extension will be used.
|
|
||||||
|
|
||||||
If you are used CMake to build LLVM, see :ref:`cmake-out-of-source-pass`.
|
|
||||||
|
|
||||||
Now that we have the build scripts set up, we just need to write the code for
|
Now that we have the build scripts set up, we just need to write the code for
|
||||||
the pass itself.
|
the pass itself.
|
||||||
|
@ -143,12 +143,12 @@ to avoid using expensive C++ runtime information.
|
||||||
|
|
||||||
.. code-block:: c++
|
.. code-block:: c++
|
||||||
|
|
||||||
bool runOnFunction(Function &F) override {
|
bool runOnFunction(Function &F) override {
|
||||||
errs() << "Hello: ";
|
errs() << "Hello: ";
|
||||||
errs().write_escaped(F.getName()) << "\n";
|
errs().write_escaped(F.getName()) << '\n';
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}; // end of struct Hello
|
}; // end of struct Hello
|
||||||
} // end of anonymous namespace
|
} // end of anonymous namespace
|
||||||
|
|
||||||
We declare a :ref:`runOnFunction <writing-an-llvm-pass-runOnFunction>` method,
|
We declare a :ref:`runOnFunction <writing-an-llvm-pass-runOnFunction>` method,
|
||||||
|
@ -180,31 +180,33 @@ As a whole, the ``.cpp`` file looks like:
|
||||||
|
|
||||||
.. code-block:: c++
|
.. code-block:: c++
|
||||||
|
|
||||||
#include "llvm/Pass.h"
|
#include "llvm/Pass.h"
|
||||||
#include "llvm/IR/Function.h"
|
#include "llvm/IR/Function.h"
|
||||||
#include "llvm/Support/raw_ostream.h"
|
#include "llvm/Support/raw_ostream.h"
|
||||||
|
|
||||||
using namespace llvm;
|
using namespace llvm;
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
struct Hello : public FunctionPass {
|
struct Hello : public FunctionPass {
|
||||||
static char ID;
|
static char ID;
|
||||||
Hello() : FunctionPass(ID) {}
|
Hello() : FunctionPass(ID) {}
|
||||||
|
|
||||||
bool runOnFunction(Function &F) override {
|
bool runOnFunction(Function &F) override {
|
||||||
errs() << "Hello: ";
|
errs() << "Hello: ";
|
||||||
errs().write_escaped(F.getName()) << '\n';
|
errs().write_escaped(F.getName()) << '\n';
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
}; // end of struct Hello
|
||||||
char Hello::ID = 0;
|
} // end of anonymous namespace
|
||||||
static RegisterPass<Hello> X("hello", "Hello World Pass", false, false);
|
|
||||||
|
char Hello::ID = 0;
|
||||||
|
static RegisterPass<Hello> X("hello", "Hello World Pass",
|
||||||
|
false /* Only looks at CFG */,
|
||||||
|
false /* Analysis Pass */);
|
||||||
|
|
||||||
Now that it's all together, compile the file with a simple "``gmake``" command
|
Now that it's all together, compile the file with a simple "``gmake``" command
|
||||||
from the top level of your build directory and you should get a new file
|
from the top level of your build directory and you should get a new file
|
||||||
"``Debug+Asserts/lib/Hello.so``". Note that everything in this file is
|
"``lib/LLVMHello.so``". Note that everything in this file is
|
||||||
contained in an anonymous namespace --- this reflects the fact that passes
|
contained in an anonymous namespace --- this reflects the fact that passes
|
||||||
are self contained units that do not need external interfaces (although they
|
are self contained units that do not need external interfaces (although they
|
||||||
can have them) to be useful.
|
can have them) to be useful.
|
||||||
|
@ -224,7 +226,7 @@ will work):
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -hello < hello.bc > /dev/null
|
$ opt -load lib/LLVMHello.so -hello < hello.bc > /dev/null
|
||||||
Hello: __main
|
Hello: __main
|
||||||
Hello: puts
|
Hello: puts
|
||||||
Hello: main
|
Hello: main
|
||||||
|
@ -241,20 +243,20 @@ To see what happened to the other string you registered, try running
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -help
|
$ opt -load lib/LLVMHello.so -help
|
||||||
OVERVIEW: llvm .bc -> .bc modular optimizer
|
OVERVIEW: llvm .bc -> .bc modular optimizer and analysis printer
|
||||||
|
|
||||||
USAGE: opt [options] <input bitcode>
|
USAGE: opt [subcommand] [options] <input bitcode file>
|
||||||
|
|
||||||
OPTIONS:
|
OPTIONS:
|
||||||
Optimizations available:
|
Optimizations available:
|
||||||
...
|
...
|
||||||
-globalopt - Global Variable Optimizer
|
-guard-widening - Widen guards
|
||||||
-globalsmodref-aa - Simple mod/ref analysis for globals
|
|
||||||
-gvn - Global Value Numbering
|
-gvn - Global Value Numbering
|
||||||
|
-gvn-hoist - Early GVN Hoisting of Expressions
|
||||||
-hello - Hello World Pass
|
-hello - Hello World Pass
|
||||||
-indvars - Induction Variable Simplification
|
-indvars - Induction Variable Simplification
|
||||||
-inline - Function Integration/Inlining
|
-inferattrs - Infer set function attributes
|
||||||
...
|
...
|
||||||
|
|
||||||
The pass name gets added as the information string for your pass, giving some
|
The pass name gets added as the information string for your pass, giving some
|
||||||
|
@ -268,21 +270,20 @@ you queue up. For example:
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -hello -time-passes < hello.bc > /dev/null
|
$ opt -load lib/LLVMHello.so -hello -time-passes < hello.bc > /dev/null
|
||||||
Hello: __main
|
Hello: __main
|
||||||
Hello: puts
|
Hello: puts
|
||||||
Hello: main
|
Hello: main
|
||||||
===============================================================================
|
===-------------------------------------------------------------------------===
|
||||||
... Pass execution timing report ...
|
... Pass execution timing report ...
|
||||||
===============================================================================
|
===-------------------------------------------------------------------------===
|
||||||
Total Execution Time: 0.02 seconds (0.0479059 wall clock)
|
Total Execution Time: 0.0007 seconds (0.0005 wall clock)
|
||||||
|
|
||||||
---User Time--- --System Time-- --User+System-- ---Wall Time--- --- Pass Name ---
|
---User Time--- --User+System-- ---Wall Time--- --- Name ---
|
||||||
0.0100 (100.0%) 0.0000 ( 0.0%) 0.0100 ( 50.0%) 0.0402 ( 84.0%) Bitcode Writer
|
0.0004 ( 55.3%) 0.0004 ( 55.3%) 0.0004 ( 75.7%) Bitcode Writer
|
||||||
0.0000 ( 0.0%) 0.0100 (100.0%) 0.0100 ( 50.0%) 0.0031 ( 6.4%) Dominator Set Construction
|
0.0003 ( 44.7%) 0.0003 ( 44.7%) 0.0001 ( 13.6%) Hello World Pass
|
||||||
0.0000 ( 0.0%) 0.0000 ( 0.0%) 0.0000 ( 0.0%) 0.0013 ( 2.7%) Module Verifier
|
0.0000 ( 0.0%) 0.0000 ( 0.0%) 0.0001 ( 10.7%) Module Verifier
|
||||||
0.0000 ( 0.0%) 0.0000 ( 0.0%) 0.0000 ( 0.0%) 0.0033 ( 6.9%) Hello World Pass
|
0.0007 (100.0%) 0.0007 (100.0%) 0.0005 (100.0%) Total
|
||||||
0.0100 (100.0%) 0.0100 (100.0%) 0.0200 (100.0%) 0.0479 (100.0%) TOTAL
|
|
||||||
|
|
||||||
As you can see, our implementation above is pretty fast. The additional
|
As you can see, our implementation above is pretty fast. The additional
|
||||||
passes listed are automatically inserted by the :program:`opt` tool to verify
|
passes listed are automatically inserted by the :program:`opt` tool to verify
|
||||||
|
@ -964,14 +965,14 @@ just does a few simple checks that don't require significant analysis to
|
||||||
compute (such as: two different globals can never alias each other, etc).
|
compute (such as: two different globals can never alias each other, etc).
|
||||||
Passes that use the `AliasAnalysis
|
Passes that use the `AliasAnalysis
|
||||||
<http://llvm.org/doxygen/classllvm_1_1AliasAnalysis.html>`_ interface (for
|
<http://llvm.org/doxygen/classllvm_1_1AliasAnalysis.html>`_ interface (for
|
||||||
example the `gcse <http://llvm.org/doxygen/structGCSE.html>`_ pass), do not
|
example the `gvn <http://llvm.org/doxygen/classllvm_1_1GVN.html>`_ pass), do not
|
||||||
care which implementation of alias analysis is actually provided, they just use
|
care which implementation of alias analysis is actually provided, they just use
|
||||||
the designated interface.
|
the designated interface.
|
||||||
|
|
||||||
From the user's perspective, commands work just like normal. Issuing the
|
From the user's perspective, commands work just like normal. Issuing the
|
||||||
command ``opt -gcse ...`` will cause the ``basicaa`` class to be instantiated
|
command ``opt -gvn ...`` will cause the ``basicaa`` class to be instantiated
|
||||||
and added to the pass sequence. Issuing the command ``opt -somefancyaa -gcse
|
and added to the pass sequence. Issuing the command ``opt -somefancyaa -gvn
|
||||||
...`` will cause the ``gcse`` pass to use the ``somefancyaa`` alias analysis
|
...`` will cause the ``gvn`` pass to use the ``somefancyaa`` alias analysis
|
||||||
(which doesn't actually exist, it's just a hypothetical example) instead.
|
(which doesn't actually exist, it's just a hypothetical example) instead.
|
||||||
|
|
||||||
.. _writing-an-llvm-pass-RegisterAnalysisGroup:
|
.. _writing-an-llvm-pass-RegisterAnalysisGroup:
|
||||||
|
@ -1092,74 +1093,69 @@ information about all of the variants of the ``--debug-pass`` option, just type
|
||||||
|
|
||||||
By using the --debug-pass=Structure option, for example, we can see how our
|
By using the --debug-pass=Structure option, for example, we can see how our
|
||||||
:ref:`Hello World <writing-an-llvm-pass-basiccode>` pass interacts with other
|
:ref:`Hello World <writing-an-llvm-pass-basiccode>` pass interacts with other
|
||||||
passes. Lets try it out with the gcse and licm passes:
|
passes. Lets try it out with the gvn and licm passes:
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -gcse -licm --debug-pass=Structure < hello.bc > /dev/null
|
$ opt -load lib/LLVMHello.so -gvn -licm --debug-pass=Structure < hello.bc > /dev/null
|
||||||
Module Pass Manager
|
ModulePass Manager
|
||||||
Function Pass Manager
|
FunctionPass Manager
|
||||||
Dominator Set Construction
|
Dominator Tree Construction
|
||||||
Immediate Dominators Construction
|
Basic Alias Analysis (stateless AA impl)
|
||||||
Global Common Subexpression Elimination
|
Function Alias Analysis Results
|
||||||
-- Immediate Dominators Construction
|
Memory Dependence Analysis
|
||||||
-- Global Common Subexpression Elimination
|
Global Value Numbering
|
||||||
Natural Loop Construction
|
Natural Loop Information
|
||||||
Loop Invariant Code Motion
|
Canonicalize natural loops
|
||||||
-- Natural Loop Construction
|
Loop-Closed SSA Form Pass
|
||||||
-- Loop Invariant Code Motion
|
Basic Alias Analysis (stateless AA impl)
|
||||||
|
Function Alias Analysis Results
|
||||||
|
Scalar Evolution Analysis
|
||||||
|
Loop Pass Manager
|
||||||
|
Loop Invariant Code Motion
|
||||||
Module Verifier
|
Module Verifier
|
||||||
-- Dominator Set Construction
|
|
||||||
-- Module Verifier
|
|
||||||
Bitcode Writer
|
Bitcode Writer
|
||||||
--Bitcode Writer
|
|
||||||
|
|
||||||
This output shows us when passes are constructed and when the analysis results
|
This output shows us when passes are constructed.
|
||||||
are known to be dead (prefixed with "``--``"). Here we see that GCSE uses
|
Here we see that GVN uses dominator tree information to do its job. The LICM pass
|
||||||
dominator and immediate dominator information to do its job. The LICM pass
|
uses natural loop information, which uses dominator tree as well.
|
||||||
uses natural loop information, which uses dominator sets, but not immediate
|
|
||||||
dominators. Because immediate dominators are no longer useful after the GCSE
|
|
||||||
pass, it is immediately destroyed. The dominator sets are then reused to
|
|
||||||
compute natural loop information, which is then used by the LICM pass.
|
|
||||||
|
|
||||||
After the LICM pass, the module verifier runs (which is automatically added by
|
After the LICM pass, the module verifier runs (which is automatically added by
|
||||||
the :program:`opt` tool), which uses the dominator set to check that the
|
the :program:`opt` tool), which uses the dominator tree to check that the
|
||||||
resultant LLVM code is well formed. After it finishes, the dominator set
|
resultant LLVM code is well formed. Note that the dominator tree is computed
|
||||||
information is destroyed, after being computed once, and shared by three
|
once, and shared by three passes.
|
||||||
passes.
|
|
||||||
|
|
||||||
Lets see how this changes when we run the :ref:`Hello World
|
Lets see how this changes when we run the :ref:`Hello World
|
||||||
<writing-an-llvm-pass-basiccode>` pass in between the two passes:
|
<writing-an-llvm-pass-basiccode>` pass in between the two passes:
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -gcse -hello -licm --debug-pass=Structure < hello.bc > /dev/null
|
$ opt -load lib/LLVMHello.so -gvn -hello -licm --debug-pass=Structure < hello.bc > /dev/null
|
||||||
Module Pass Manager
|
ModulePass Manager
|
||||||
Function Pass Manager
|
FunctionPass Manager
|
||||||
Dominator Set Construction
|
Dominator Tree Construction
|
||||||
Immediate Dominators Construction
|
Basic Alias Analysis (stateless AA impl)
|
||||||
Global Common Subexpression Elimination
|
Function Alias Analysis Results
|
||||||
-- Dominator Set Construction
|
Memory Dependence Analysis
|
||||||
-- Immediate Dominators Construction
|
Global Value Numbering
|
||||||
-- Global Common Subexpression Elimination
|
|
||||||
Hello World Pass
|
Hello World Pass
|
||||||
-- Hello World Pass
|
Dominator Tree Construction
|
||||||
Dominator Set Construction
|
Natural Loop Information
|
||||||
Natural Loop Construction
|
Canonicalize natural loops
|
||||||
Loop Invariant Code Motion
|
Loop-Closed SSA Form Pass
|
||||||
-- Natural Loop Construction
|
Basic Alias Analysis (stateless AA impl)
|
||||||
-- Loop Invariant Code Motion
|
Function Alias Analysis Results
|
||||||
|
Scalar Evolution Analysis
|
||||||
|
Loop Pass Manager
|
||||||
|
Loop Invariant Code Motion
|
||||||
Module Verifier
|
Module Verifier
|
||||||
-- Dominator Set Construction
|
|
||||||
-- Module Verifier
|
|
||||||
Bitcode Writer
|
Bitcode Writer
|
||||||
--Bitcode Writer
|
|
||||||
Hello: __main
|
Hello: __main
|
||||||
Hello: puts
|
Hello: puts
|
||||||
Hello: main
|
Hello: main
|
||||||
|
|
||||||
Here we see that the :ref:`Hello World <writing-an-llvm-pass-basiccode>` pass
|
Here we see that the :ref:`Hello World <writing-an-llvm-pass-basiccode>` pass
|
||||||
has killed the Dominator Set pass, even though it doesn't modify the code at
|
has killed the Dominator Tree pass, even though it doesn't modify the code at
|
||||||
all! To fix this, we need to add the following :ref:`getAnalysisUsage
|
all! To fix this, we need to add the following :ref:`getAnalysisUsage
|
||||||
<writing-an-llvm-pass-getAnalysisUsage>` method to our pass:
|
<writing-an-llvm-pass-getAnalysisUsage>` method to our pass:
|
||||||
|
|
||||||
|
@ -1174,26 +1170,26 @@ Now when we run our pass, we get this output:
|
||||||
|
|
||||||
.. code-block:: console
|
.. code-block:: console
|
||||||
|
|
||||||
$ opt -load ../../Debug+Asserts/lib/Hello.so -gcse -hello -licm --debug-pass=Structure < hello.bc > /dev/null
|
$ opt -load lib/LLVMHello.so -gvn -hello -licm --debug-pass=Structure < hello.bc > /dev/null
|
||||||
Pass Arguments: -gcse -hello -licm
|
Pass Arguments: -gvn -hello -licm
|
||||||
Module Pass Manager
|
ModulePass Manager
|
||||||
Function Pass Manager
|
FunctionPass Manager
|
||||||
Dominator Set Construction
|
Dominator Tree Construction
|
||||||
Immediate Dominators Construction
|
Basic Alias Analysis (stateless AA impl)
|
||||||
Global Common Subexpression Elimination
|
Function Alias Analysis Results
|
||||||
-- Immediate Dominators Construction
|
Memory Dependence Analysis
|
||||||
-- Global Common Subexpression Elimination
|
Global Value Numbering
|
||||||
Hello World Pass
|
Hello World Pass
|
||||||
-- Hello World Pass
|
Natural Loop Information
|
||||||
Natural Loop Construction
|
Canonicalize natural loops
|
||||||
Loop Invariant Code Motion
|
Loop-Closed SSA Form Pass
|
||||||
-- Loop Invariant Code Motion
|
Basic Alias Analysis (stateless AA impl)
|
||||||
-- Natural Loop Construction
|
Function Alias Analysis Results
|
||||||
|
Scalar Evolution Analysis
|
||||||
|
Loop Pass Manager
|
||||||
|
Loop Invariant Code Motion
|
||||||
Module Verifier
|
Module Verifier
|
||||||
-- Dominator Set Construction
|
|
||||||
-- Module Verifier
|
|
||||||
Bitcode Writer
|
Bitcode Writer
|
||||||
--Bitcode Writer
|
|
||||||
Hello: __main
|
Hello: __main
|
||||||
Hello: puts
|
Hello: puts
|
||||||
Hello: main
|
Hello: main
|
||||||
|
|
Loading…
Reference in New Issue