annotate llvm/docs/tutorial/MyFirstLanguageFrontend/LangImpl08.rst @ 266:00f31e85ec16 default tip

Added tag current for changeset 31d058e83c98
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Sat, 14 Oct 2023 10:13:55 +0900
parents 1f2b6ac9f198
children
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 ========================================
anatofuz
parents:
diff changeset
2 Kaleidoscope: Compiling to Object Code
anatofuz
parents:
diff changeset
3 ========================================
anatofuz
parents:
diff changeset
4
anatofuz
parents:
diff changeset
5 .. contents::
anatofuz
parents:
diff changeset
6 :local:
anatofuz
parents:
diff changeset
7
anatofuz
parents:
diff changeset
8 Chapter 8 Introduction
anatofuz
parents:
diff changeset
9 ======================
anatofuz
parents:
diff changeset
10
anatofuz
parents:
diff changeset
11 Welcome to Chapter 8 of the "`Implementing a language with LLVM
anatofuz
parents:
diff changeset
12 <index.html>`_" tutorial. This chapter describes how to compile our
anatofuz
parents:
diff changeset
13 language down to object files.
anatofuz
parents:
diff changeset
14
anatofuz
parents:
diff changeset
15 Choosing a target
anatofuz
parents:
diff changeset
16 =================
anatofuz
parents:
diff changeset
17
anatofuz
parents:
diff changeset
18 LLVM has native support for cross-compilation. You can compile to the
anatofuz
parents:
diff changeset
19 architecture of your current machine, or just as easily compile for
anatofuz
parents:
diff changeset
20 other architectures. In this tutorial, we'll target the current
anatofuz
parents:
diff changeset
21 machine.
anatofuz
parents:
diff changeset
22
anatofuz
parents:
diff changeset
23 To specify the architecture that you want to target, we use a string
anatofuz
parents:
diff changeset
24 called a "target triple". This takes the form
anatofuz
parents:
diff changeset
25 ``<arch><sub>-<vendor>-<sys>-<abi>`` (see the `cross compilation docs
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
26 <https://clang.llvm.org/docs/CrossCompilation.html#target-triple>`_).
150
anatofuz
parents:
diff changeset
27
anatofuz
parents:
diff changeset
28 As an example, we can see what clang thinks is our current target
anatofuz
parents:
diff changeset
29 triple:
anatofuz
parents:
diff changeset
30
anatofuz
parents:
diff changeset
31 ::
anatofuz
parents:
diff changeset
32
anatofuz
parents:
diff changeset
33 $ clang --version | grep Target
anatofuz
parents:
diff changeset
34 Target: x86_64-unknown-linux-gnu
anatofuz
parents:
diff changeset
35
anatofuz
parents:
diff changeset
36 Running this command may show something different on your machine as
anatofuz
parents:
diff changeset
37 you might be using a different architecture or operating system to me.
anatofuz
parents:
diff changeset
38
anatofuz
parents:
diff changeset
39 Fortunately, we don't need to hard-code a target triple to target the
anatofuz
parents:
diff changeset
40 current machine. LLVM provides ``sys::getDefaultTargetTriple``, which
anatofuz
parents:
diff changeset
41 returns the target triple of the current machine.
anatofuz
parents:
diff changeset
42
anatofuz
parents:
diff changeset
43 .. code-block:: c++
anatofuz
parents:
diff changeset
44
anatofuz
parents:
diff changeset
45 auto TargetTriple = sys::getDefaultTargetTriple();
anatofuz
parents:
diff changeset
46
anatofuz
parents:
diff changeset
47 LLVM doesn't require us to link in all the target
anatofuz
parents:
diff changeset
48 functionality. For example, if we're just using the JIT, we don't need
anatofuz
parents:
diff changeset
49 the assembly printers. Similarly, if we're only targeting certain
anatofuz
parents:
diff changeset
50 architectures, we can only link in the functionality for those
anatofuz
parents:
diff changeset
51 architectures.
anatofuz
parents:
diff changeset
52
anatofuz
parents:
diff changeset
53 For this example, we'll initialize all the targets for emitting object
anatofuz
parents:
diff changeset
54 code.
anatofuz
parents:
diff changeset
55
anatofuz
parents:
diff changeset
56 .. code-block:: c++
anatofuz
parents:
diff changeset
57
anatofuz
parents:
diff changeset
58 InitializeAllTargetInfos();
anatofuz
parents:
diff changeset
59 InitializeAllTargets();
anatofuz
parents:
diff changeset
60 InitializeAllTargetMCs();
anatofuz
parents:
diff changeset
61 InitializeAllAsmParsers();
anatofuz
parents:
diff changeset
62 InitializeAllAsmPrinters();
anatofuz
parents:
diff changeset
63
anatofuz
parents:
diff changeset
64 We can now use our target triple to get a ``Target``:
anatofuz
parents:
diff changeset
65
anatofuz
parents:
diff changeset
66 .. code-block:: c++
anatofuz
parents:
diff changeset
67
anatofuz
parents:
diff changeset
68 std::string Error;
anatofuz
parents:
diff changeset
69 auto Target = TargetRegistry::lookupTarget(TargetTriple, Error);
anatofuz
parents:
diff changeset
70
anatofuz
parents:
diff changeset
71 // Print an error and exit if we couldn't find the requested target.
anatofuz
parents:
diff changeset
72 // This generally occurs if we've forgotten to initialise the
anatofuz
parents:
diff changeset
73 // TargetRegistry or we have a bogus target triple.
anatofuz
parents:
diff changeset
74 if (!Target) {
anatofuz
parents:
diff changeset
75 errs() << Error;
anatofuz
parents:
diff changeset
76 return 1;
anatofuz
parents:
diff changeset
77 }
anatofuz
parents:
diff changeset
78
anatofuz
parents:
diff changeset
79 Target Machine
anatofuz
parents:
diff changeset
80 ==============
anatofuz
parents:
diff changeset
81
anatofuz
parents:
diff changeset
82 We will also need a ``TargetMachine``. This class provides a complete
anatofuz
parents:
diff changeset
83 machine description of the machine we're targeting. If we want to
anatofuz
parents:
diff changeset
84 target a specific feature (such as SSE) or a specific CPU (such as
anatofuz
parents:
diff changeset
85 Intel's Sandylake), we do so now.
anatofuz
parents:
diff changeset
86
anatofuz
parents:
diff changeset
87 To see which features and CPUs that LLVM knows about, we can use
anatofuz
parents:
diff changeset
88 ``llc``. For example, let's look at x86:
anatofuz
parents:
diff changeset
89
anatofuz
parents:
diff changeset
90 ::
anatofuz
parents:
diff changeset
91
anatofuz
parents:
diff changeset
92 $ llvm-as < /dev/null | llc -march=x86 -mattr=help
anatofuz
parents:
diff changeset
93 Available CPUs for this target:
anatofuz
parents:
diff changeset
94
anatofuz
parents:
diff changeset
95 amdfam10 - Select the amdfam10 processor.
anatofuz
parents:
diff changeset
96 athlon - Select the athlon processor.
anatofuz
parents:
diff changeset
97 athlon-4 - Select the athlon-4 processor.
anatofuz
parents:
diff changeset
98 ...
anatofuz
parents:
diff changeset
99
anatofuz
parents:
diff changeset
100 Available features for this target:
anatofuz
parents:
diff changeset
101
anatofuz
parents:
diff changeset
102 16bit-mode - 16-bit mode (i8086).
anatofuz
parents:
diff changeset
103 32bit-mode - 32-bit mode (80386).
anatofuz
parents:
diff changeset
104 3dnow - Enable 3DNow! instructions.
anatofuz
parents:
diff changeset
105 3dnowa - Enable 3DNow! Athlon instructions.
anatofuz
parents:
diff changeset
106 ...
anatofuz
parents:
diff changeset
107
anatofuz
parents:
diff changeset
108 For our example, we'll use the generic CPU without any additional
anatofuz
parents:
diff changeset
109 features, options or relocation model.
anatofuz
parents:
diff changeset
110
anatofuz
parents:
diff changeset
111 .. code-block:: c++
anatofuz
parents:
diff changeset
112
anatofuz
parents:
diff changeset
113 auto CPU = "generic";
anatofuz
parents:
diff changeset
114 auto Features = "";
anatofuz
parents:
diff changeset
115
anatofuz
parents:
diff changeset
116 TargetOptions opt;
252
1f2b6ac9f198 LLVM16-1
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 236
diff changeset
117 auto RM = std::optional<Reloc::Model>();
150
anatofuz
parents:
diff changeset
118 auto TargetMachine = Target->createTargetMachine(TargetTriple, CPU, Features, opt, RM);
anatofuz
parents:
diff changeset
119
anatofuz
parents:
diff changeset
120
anatofuz
parents:
diff changeset
121 Configuring the Module
anatofuz
parents:
diff changeset
122 ======================
anatofuz
parents:
diff changeset
123
anatofuz
parents:
diff changeset
124 We're now ready to configure our module, to specify the target and
anatofuz
parents:
diff changeset
125 data layout. This isn't strictly necessary, but the `frontend
anatofuz
parents:
diff changeset
126 performance guide <../Frontend/PerformanceTips.html>`_ recommends
anatofuz
parents:
diff changeset
127 this. Optimizations benefit from knowing about the target and data
anatofuz
parents:
diff changeset
128 layout.
anatofuz
parents:
diff changeset
129
anatofuz
parents:
diff changeset
130 .. code-block:: c++
anatofuz
parents:
diff changeset
131
anatofuz
parents:
diff changeset
132 TheModule->setDataLayout(TargetMachine->createDataLayout());
236
c4bab56944e8 LLVM 16
kono
parents: 173
diff changeset
133 TheModule->setTargetTriple(TargetTriple);
c4bab56944e8 LLVM 16
kono
parents: 173
diff changeset
134
150
anatofuz
parents:
diff changeset
135 Emit Object Code
anatofuz
parents:
diff changeset
136 ================
anatofuz
parents:
diff changeset
137
anatofuz
parents:
diff changeset
138 We're ready to emit object code! Let's define where we want to write
anatofuz
parents:
diff changeset
139 our file to:
anatofuz
parents:
diff changeset
140
anatofuz
parents:
diff changeset
141 .. code-block:: c++
anatofuz
parents:
diff changeset
142
anatofuz
parents:
diff changeset
143 auto Filename = "output.o";
anatofuz
parents:
diff changeset
144 std::error_code EC;
anatofuz
parents:
diff changeset
145 raw_fd_ostream dest(Filename, EC, sys::fs::OF_None);
anatofuz
parents:
diff changeset
146
anatofuz
parents:
diff changeset
147 if (EC) {
anatofuz
parents:
diff changeset
148 errs() << "Could not open file: " << EC.message();
anatofuz
parents:
diff changeset
149 return 1;
anatofuz
parents:
diff changeset
150 }
anatofuz
parents:
diff changeset
151
anatofuz
parents:
diff changeset
152 Finally, we define a pass that emits object code, then we run that
anatofuz
parents:
diff changeset
153 pass:
anatofuz
parents:
diff changeset
154
anatofuz
parents:
diff changeset
155 .. code-block:: c++
anatofuz
parents:
diff changeset
156
anatofuz
parents:
diff changeset
157 legacy::PassManager pass;
anatofuz
parents:
diff changeset
158 auto FileType = CGFT_ObjectFile;
anatofuz
parents:
diff changeset
159
anatofuz
parents:
diff changeset
160 if (TargetMachine->addPassesToEmitFile(pass, dest, nullptr, FileType)) {
anatofuz
parents:
diff changeset
161 errs() << "TargetMachine can't emit a file of this type";
anatofuz
parents:
diff changeset
162 return 1;
anatofuz
parents:
diff changeset
163 }
anatofuz
parents:
diff changeset
164
anatofuz
parents:
diff changeset
165 pass.run(*TheModule);
anatofuz
parents:
diff changeset
166 dest.flush();
anatofuz
parents:
diff changeset
167
anatofuz
parents:
diff changeset
168 Putting It All Together
anatofuz
parents:
diff changeset
169 =======================
anatofuz
parents:
diff changeset
170
anatofuz
parents:
diff changeset
171 Does it work? Let's give it a try. We need to compile our code, but
anatofuz
parents:
diff changeset
172 note that the arguments to ``llvm-config`` are different to the previous chapters.
anatofuz
parents:
diff changeset
173
anatofuz
parents:
diff changeset
174 ::
anatofuz
parents:
diff changeset
175
anatofuz
parents:
diff changeset
176 $ clang++ -g -O3 toy.cpp `llvm-config --cxxflags --ldflags --system-libs --libs all` -o toy
anatofuz
parents:
diff changeset
177
anatofuz
parents:
diff changeset
178 Let's run it, and define a simple ``average`` function. Press Ctrl-D
anatofuz
parents:
diff changeset
179 when you're done.
anatofuz
parents:
diff changeset
180
anatofuz
parents:
diff changeset
181 ::
236
c4bab56944e8 LLVM 16
kono
parents: 173
diff changeset
182
150
anatofuz
parents:
diff changeset
183 $ ./toy
anatofuz
parents:
diff changeset
184 ready> def average(x y) (x + y) * 0.5;
anatofuz
parents:
diff changeset
185 ^D
anatofuz
parents:
diff changeset
186 Wrote output.o
anatofuz
parents:
diff changeset
187
anatofuz
parents:
diff changeset
188 We have an object file! To test it, let's write a simple program and
anatofuz
parents:
diff changeset
189 link it with our output. Here's the source code:
anatofuz
parents:
diff changeset
190
anatofuz
parents:
diff changeset
191 .. code-block:: c++
anatofuz
parents:
diff changeset
192
anatofuz
parents:
diff changeset
193 #include <iostream>
anatofuz
parents:
diff changeset
194
anatofuz
parents:
diff changeset
195 extern "C" {
anatofuz
parents:
diff changeset
196 double average(double, double);
anatofuz
parents:
diff changeset
197 }
anatofuz
parents:
diff changeset
198
anatofuz
parents:
diff changeset
199 int main() {
anatofuz
parents:
diff changeset
200 std::cout << "average of 3.0 and 4.0: " << average(3.0, 4.0) << std::endl;
anatofuz
parents:
diff changeset
201 }
anatofuz
parents:
diff changeset
202
anatofuz
parents:
diff changeset
203 We link our program to output.o and check the result is what we
anatofuz
parents:
diff changeset
204 expected:
anatofuz
parents:
diff changeset
205
anatofuz
parents:
diff changeset
206 ::
anatofuz
parents:
diff changeset
207
anatofuz
parents:
diff changeset
208 $ clang++ main.cpp output.o -o main
anatofuz
parents:
diff changeset
209 $ ./main
anatofuz
parents:
diff changeset
210 average of 3.0 and 4.0: 3.5
anatofuz
parents:
diff changeset
211
anatofuz
parents:
diff changeset
212 Full Code Listing
anatofuz
parents:
diff changeset
213 =================
anatofuz
parents:
diff changeset
214
anatofuz
parents:
diff changeset
215 .. literalinclude:: ../../../examples/Kaleidoscope/Chapter8/toy.cpp
anatofuz
parents:
diff changeset
216 :language: c++
anatofuz
parents:
diff changeset
217
anatofuz
parents:
diff changeset
218 `Next: Adding Debug Information <LangImpl09.html>`_