-
Notifications
You must be signed in to change notification settings - Fork 200
/
Copy pathbuild_llvm.sh
319 lines (295 loc) · 13.3 KB
/
build_llvm.sh
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
#!/bin/bash
# ============================================================================ #
# Copyright (c) 2022 - 2025 NVIDIA Corporation & Affiliates. #
# All rights reserved. #
# #
# This source code and the accompanying materials are made available under #
# the terms of the Apache License 2.0 which accompanies this distribution. #
# ============================================================================ #
# This scripts builds the clang and mlir project from the source in the LLVM submodule.
# The binaries will be installed in the folder defined by the LLVM_INSTALL_PREFIX environment
# variable, or in $HOME/.llvm if LLVM_INSTALL_PREFIX is not defined.
# If Python bindings are generated, pybind11 will be built and installed in the location
# defined by PYBIND11_INSTALL_PREFIX unless that folder already exists.
#
# Usage:
# bash scripts/build_llvm.sh
# -or-
# bash scripts/build_llvm.sh -c Debug
# -or-
# LLVM_INSTALL_PREFIX=/installation/path/ bash scripts/build_llvm.sh
#
# For documentation on how to assemble a complete toolchain, multi-stage builds,
# and OpenMP support within Clang, see
# - https://clang.llvm.org/docs/Toolchain.html
# - https://llvm.org/docs/AdvancedBuilds.html
# - https://github.com/llvm/llvm-project/tree/main/clang/cmake/caches
# - https://github.com/llvm/llvm-project/blob/main/openmp/docs/SupportAndFAQ.rst#q-how-to-build-an-openmp-gpu-offload-capable-compiler
#
# Not directly relevant, but still interesting to know:
# - https://www.linaro.org/blog/building-glibc-with-llvm-the-how-and-why/
# - https://www.collabora.com/news-and-blog/blog/2023/01/17/a-brave-new-world-building-glibc-with-llvm/
LLVM_INSTALL_PREFIX=${LLVM_INSTALL_PREFIX:-$HOME/.llvm}
LLVM_PROJECTS=${LLVM_PROJECTS:-'clang;lld;mlir;python-bindings'}
PYBIND11_INSTALL_PREFIX=${PYBIND11_INSTALL_PREFIX:-/usr/local/pybind11}
Python3_EXECUTABLE=${Python3_EXECUTABLE:-python3}
# Process command line arguments.
build_configuration=Release
verbose=false
__optind__=$OPTIND
OPTIND=1
while getopts ":c:j:k:v" opt; do
case $opt in
c) build_configuration="$OPTARG"
;;
j) build_concurrency="-j $OPTARG"
;;
k) ninja_keep_going="-k $OPTARG"
;;
v) verbose=true
;;
:) echo "Option -$OPTARG requires an argument."
(return 0 2>/dev/null) && return 1 || exit 1
;;
\?) echo "Invalid command line option -$OPTARG" >&2
(return 0 2>/dev/null) && return 1 || exit 1
;;
esac
done
OPTIND=$__optind__
working_dir=`pwd`
this_file_dir=`dirname "$(readlink -f "${BASH_SOURCE[0]}")"`
echo "Configured C compiler: $CC"
echo "Configured C++ compiler: $CXX"
# Check if we build python bindings and build pybind11 from source if necessary.
projects=(`echo $LLVM_PROJECTS | tr ';' ' '`)
llvm_projects=`printf "%s;" "${projects[@]}"`
if [ -z "${llvm_projects##*python-bindings;*}" ]; then
mlir_python_bindings=ON
projects=("${projects[@]/python-bindings}")
if [ ! -d "$PYBIND11_INSTALL_PREFIX" ] || [ -z "$(ls -A "$PYBIND11_INSTALL_PREFIX"/* 2> /dev/null)" ]; then
cd "$this_file_dir" && cd $(git rev-parse --show-toplevel)
echo "Building PyBind11..."
git submodule update --init --recursive --recommend-shallow --single-branch tpls/pybind11
mkdir "tpls/pybind11/build" && cd "tpls/pybind11/build"
cmake -G Ninja ../ -DCMAKE_INSTALL_PREFIX="$PYBIND11_INSTALL_PREFIX" -DPYBIND11_TEST=False
cmake --build . --target install --config Release
fi
fi
# Prepare the source and build directory.
if [ ! -d "$LLVM_SOURCE" ] || [ -z "$(ls -A "$LLVM_SOURCE"/* 2> /dev/null)" ]; then
echo "Cloning LLVM submodule..."
cd "$this_file_dir" && cd $(git rev-parse --show-toplevel)
LLVM_SOURCE="${LLVM_SOURCE:-$HOME/.llvm-project}"
llvm_repo="$(git config --file=.gitmodules submodule.tpls/llvm.url)"
llvm_commit="$(git submodule | grep tpls/llvm | cut -c2- | cut -d ' ' -f1)"
git clone --filter=tree:0 "$llvm_repo" "$LLVM_SOURCE"
cd "$LLVM_SOURCE" && git checkout $llvm_commit
LLVM_CMAKE_PATCHES=${LLVM_CMAKE_PATCHES:-"$this_file_dir/../tpls/customizations/llvm"}
if [ -d "$LLVM_CMAKE_PATCHES" ]; then
echo "Applying LLVM patches in $LLVM_CMAKE_PATCHES..."
for patch in `find "$LLVM_CMAKE_PATCHES"/* -maxdepth 0 -type f -name '*.diff'`; do
# Check if patch is already applied.
git apply "$patch" --ignore-whitespace --reverse --check 2>/dev/null
if [ ! 0 -eq $? ]; then
# If the patch is not yet applied, apply the patch.
git apply "$patch" --ignore-whitespace
if [ ! 0 -eq $? ]; then
echo "Applying patch $patch failed. Please update patch."
(return 0 2>/dev/null) && return 1 || exit 1
else
echo "Applied patch $patch."
fi
fi
done
fi
fi
llvm_build_dir="$LLVM_SOURCE/${LLVM_BUILD_FOLDER:-build}"
llvm_log_dir="$llvm_build_dir/logs"
mkdir -p "$LLVM_INSTALL_PREFIX"
mkdir -p "$llvm_build_dir" && cd "$llvm_build_dir"
mkdir -p "$llvm_log_dir" && rm -rf "$llvm_log_dir"/*
# Specify which components we need to keep the size of the LLVM build down.
# To get a list of install targets, check the output of the following command
# in the build folder:
# ninja -t targets | grep -Po 'install-\K.*(?=-stripped:)'
echo "Preparing LLVM build..."
if [ -z "${llvm_projects##*runtimes;*}" ]; then
echo "- including runtime components"
llvm_runtimes+="libcxx;libcxxabi;libunwind;compiler-rt;"
projects=("${projects[@]/runtimes}")
projects=("${projects[@]/libcxx}")
projects=("${projects[@]/libcxxabi}")
projects=("${projects[@]/libunwind}")
projects=("${projects[@]/compiler-rt}")
elif [ -z "${llvm_projects##*compiler-rt;*}" ]; then
echo "- including compiler-rt"
llvm_runtimes+="compiler-rt;"
projects=("${projects[@]/compiler-rt}")
fi
llvm_projects=`printf "%s;" "${projects[@]}"`
if [ -z "${llvm_projects##*clang;*}" ]; then
echo "- including Clang components"
llvm_components+="clang;clang-format;clang-cmake-exports;clang-headers;clang-libraries;clang-resource-headers;"
projects=("${projects[@]/clang}")
fi
if [ -z "${llvm_projects##*flang;*}" ]; then
echo "- including Flang components"
llvm_components+="flang-new;"
projects=("${projects[@]/flang}")
fi
if [ -z "${llvm_projects##*openmp;*}" ]; then
echo "- including OpenMP components"
# There are no suitable distribution components for libomp.
# We instead manually build suitable targets.
install_targets+=" omp"
llvm_components+="omptarget;openmp-resource-headers;"
projects=("${projects[@]/openmp}")
fi
if [ -z "${llvm_projects##*mlir;*}" ]; then
echo "- including MLIR components"
llvm_components+="mlir-cmake-exports;mlir-headers;mlir-libraries;mlir-tblgen;"
projects=("${projects[@]/mlir}")
if [ "$mlir_python_bindings" == "ON" ]; then
echo "- including MLIR Python bindings"
llvm_components+="MLIRPythonModules;mlir-python-sources;"
fi
fi
if [ -z "${llvm_projects##*lld;*}" ]; then
echo "- including LLD components"
llvm_enable_zlib=ON # certain system libraries are compressed with ELFCOMPRESS_ZLIB, requiring zlib support for lld
llvm_components+="lld;"
projects=("${projects[@]/lld}")
fi
echo "- including general tools and components"
llvm_components+="cmake-exports;llvm-headers;llvm-libraries;"
llvm_components+="llvm-config;llc;llvm-ar;llvm-as;llvm-nm;llvm-symbolizer;llvm-profdata;llvm-cov;"
llvm_components+="FileCheck;count;not;"
if [ "$(echo ${projects[@]} | xargs)" != "" ]; then
echo "- including additional project(s) "$(echo "${projects[*]}" | xargs | tr ' ' ',')
unset llvm_components
install_targets="install $install_targets"
else
install_targets="install-distribution-stripped $install_targets"
if [ -n "$mlir_python_bindings" ]; then
# Cherry-pick the necessary commit to have a distribution target
# for the mlir-python-sources; to be removed after we update to LLVM 17.
echo "Cherry-picking commit 9494bd84df3c5b496fc087285af9ff40d7859b6a"
git cherry-pick --no-commit 9494bd84df3c5b496fc087285af9ff40d7859b6a
if [ ! 0 -eq $? ]; then
echo "Cherry-pick failed."
if $(git rev-parse --is-shallow-repository); then
echo "Unshallow the repository and try again."
(return 0 2>/dev/null) && return 1 || exit 1
fi
fi
fi
fi
# A hack, since otherwise the build can fail due to line endings in the LLVM script:
cat "$LLVM_SOURCE/llvm/cmake/config.guess" | tr -d '\r' > ~config.guess
cat ~config.guess > "$LLVM_SOURCE/llvm/cmake/config.guess" && rm -rf ~config.guess
# Some flags that may be useful to build a GPU-offload-capable compiler:
# targets_to_build="host;NVPTX"
# -DLLVM_TARGETS_TO_BUILD='"$targets_to_build"' \
# -DLIBOMPTARGET_DEVICE_ARCHITECTURES=sm_70,sm_75,sm_80
# maybe: -DLLVM_RUNTIME_TARGETS='nvptx64-nvidia-cuda' \
cmake_args=" \
-DLLVM_DEFAULT_TARGET_TRIPLE='"$(bash $LLVM_SOURCE/llvm/cmake/config.guess)"' \
-DCMAKE_BUILD_TYPE=$build_configuration \
-DCMAKE_INSTALL_PREFIX='"$LLVM_INSTALL_PREFIX"' \
-DLLVM_ENABLE_PROJECTS='"${llvm_projects%;}"' \
-DLLVM_ENABLE_RUNTIMES='"${llvm_runtimes%;}"' \
-DLLVM_DISTRIBUTION_COMPONENTS='"${llvm_components%;}"' \
-DLLVM_ENABLE_ZLIB=${llvm_enable_zlib:-OFF} \
-DZLIB_ROOT='"$ZLIB_INSTALL_PREFIX"' \
-DPython3_EXECUTABLE='"$Python3_EXECUTABLE"' \
-DMLIR_ENABLE_BINDINGS_PYTHON=$mlir_python_bindings \
-DCMAKE_EXPORT_COMPILE_COMMANDS=ON \
-DCMAKE_CXX_FLAGS='-w'"
if [ -z "$LLVM_CMAKE_CACHE" ]; then
LLVM_CMAKE_CACHE=`find "$this_file_dir/.." -path '*/cmake/caches/*' -name LLVM.cmake`
fi
if [ -f "$LLVM_CMAKE_CACHE" ]; then
echo "Using CMake cache in $LLVM_CMAKE_CACHE."
cp "$LLVM_CMAKE_CACHE" custom_toolchain.cmake
cmake_cache='-C custom_toolchain.cmake'
# Note on combining a CMake cache with command line definitions:
# If a set(... CACHE ...) call in the -C file does not use FORCE,
# the command line define takes precedence regardless of order.
# If set(... CACHE ... FORCE) is used, the order of definition
# matters and the last defined value is used.
# See https://cmake.org/cmake/help/latest/manual/cmake.1.html.
else
echo "No CMake file found to populate the initial cache with. Set LLVM_CMAKE_CACHE to define one."
fi
# Generate CMake files.
if $verbose; then
echo $cmake_args $cmake_cache | xargs cmake -G Ninja $LLVM_SOURCE/llvm
else
echo $cmake_args $cmake_cache | xargs cmake -G Ninja $LLVM_SOURCE/llvm \
2> "$llvm_log_dir/cmake_error.txt" 1> "$llvm_log_dir/cmake_output.txt"
fi
# Build and install the defined distribution.
echo "Building LLVM with configuration $build_configuration..."
if $verbose; then
ninja $ninja_keep_going $build_concurrency $install_targets
status=$?
else
echo "The progress of the build is being logged to $llvm_log_dir/ninja_output.txt."
ninja $ninja_keep_going $build_concurrency $install_targets \
2> "$llvm_log_dir/ninja_error.txt" 1> "$llvm_log_dir/ninja_output.txt"
status=$?
fi
if [ "$status" = "" ] || [ ! "$status" -eq "0" ]; then
echo "Failed to build compiler components. Please check the files in the $llvm_log_dir/logs directory."
cd "$working_dir" && (return 0 2>/dev/null) && return 1 || exit 1
else
cp bin/llvm-lit "$LLVM_INSTALL_PREFIX/bin/"
fi
# Build and install runtimes using the newly built toolchain.
if [ -n "$llvm_runtimes" ]; then
echo "Building runtime components..."
if $verbose; then
ninja install-runtimes
status=$?
else
ninja install-runtimes 2>> "$llvm_log_dir/ninja_error.txt" 1>> "$llvm_log_dir/ninja_output.txt"
status=$?
fi
if [ "$status" = "" ] || [ ! "$status" -eq "0" ]; then
echo "Failed to build runtime components. Please check the files in the $llvm_log_dir/logs directory."
cd "$working_dir" && (return 0 2>/dev/null) && return 1 || exit 1
else
# Depending on the exact build configuration,
# no install step is defined for builtins when compiler-rt is built
# as runtime rather than as project. Similarly, no install step is
# defined for libomp if we install a distribution.
# Invoking the installation manually for these.
if $verbose; then
cmake -P runtimes/builtins-bins/cmake_install.cmake
if [ -n "$(echo $install_targets | grep omp)" ]; then
cmake -P projects/openmp/runtime/src/cmake_install.cmake
fi
else
cmake -P runtimes/builtins-bins/cmake_install.cmake \
2>> "$llvm_log_dir/ninja_error.txt" 1>> "$llvm_log_dir/ninja_output.txt"
if [ -n "$(echo $install_targets | grep omp)" ]; then
cmake -P projects/openmp/runtime/src/cmake_install.cmake \
2>> "$llvm_log_dir/ninja_error.txt" 1>> "$llvm_log_dir/ninja_output.txt"
fi
fi
echo "Successfully added runtime components $(echo ${llvm_runtimes%;} | sed 's/;/, /g')."
# We can use a default config file to set specific clang configurations.
# See https://clang.llvm.org/docs/UsersManual.html#configuration-files
clang_config_file="$LLVM_INSTALL_PREFIX/bin/clang++.cfg"
echo '-L"'$LLVM_INSTALL_PREFIX/lib'"' > "$clang_config_file"
echo '-Wl,-rpath,"'$LLVM_INSTALL_PREFIX/lib'"' >> "$clang_config_file"
target_specific_libs=`ls -d "$LLVM_INSTALL_PREFIX/lib"/*linux*`
for libdir in $target_specific_libs; do
echo '-L"'$libdir'"' >> "$clang_config_file"
echo '-Wl,-rpath,"'$libdir'"' >> "$clang_config_file"
done
echo "Added default configuration $clang_config_file."
fi
fi
cd "$working_dir" && echo "Installed llvm build in directory: $LLVM_INSTALL_PREFIX"