forked from microsoft/onnxruntime-genai
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbuild.py
472 lines (367 loc) · 16.5 KB
/
build.py
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
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
# Copyright (c) Microsoft Corporation. All rights reserved.
# Licensed under the MIT License.
from __future__ import annotations
import argparse
import os
import platform
import shutil
import sys
import warnings
from pathlib import Path
REPO_ROOT = Path(__file__).parent
sys.path.append(str(REPO_ROOT / "tools" / "python"))
import util # ./tools/python/util noqa: E402
log = util.get_logger("build.py")
def _path_from_env_var(env_var: str):
env_var_value = os.environ.get(env_var)
return Path(env_var_value) if env_var_value is not None else None
def _parse_args():
parser = argparse.ArgumentParser(
description="ONNX Runtime GenAI Build Driver.",
# files containing arguments can be specified on the command line with "@<filename>" and the arguments within
# will be included at that point
fromfile_prefix_chars="@",
formatter_class=argparse.ArgumentDefaultsHelpFormatter,
)
parser.add_argument(
"--build_dir",
type=Path,
# We set the default programmatically as it needs to take into account whether we're cross-compiling
help="Path to the build directory. Defaults to 'build/<target platform>'. "
"The build configuration will be a subdirectory of the build directory. e.g. build/Linux/Debug",
)
parser.add_argument(
"--config",
default="RelWithDebInfo",
type=str,
choices=["Debug", "MinSizeRel", "Release", "RelWithDebInfo"],
help="Configuration to build.")
# Build phases.
parser.add_argument("--update", action="store_true", help="Update makefiles.")
parser.add_argument("--build", action="store_true", help="Build.")
parser.add_argument("--test", action="store_true", help="Run tests.")
parser.add_argument(
"--clean", action="store_true", help="Run 'cmake --build --target clean' for the selected config/s."
)
parser.add_argument("--skip_tests", action="store_true", help="Skip all tests. Overrides --test.")
parser.add_argument("--skip_wheel", action="store_true", help="Skip building the Python wheel.")
parser.add_argument("--skip_csharp", action="store_true", help="Skip building the C# API.")
# Default to not building the Java bindings
parser.add_argument("--build_java", action="store_true", help="Build Java bindings.")
parser.add_argument("--parallel", action="store_true", help="Enable parallel build.")
# CI's sometimes explicitly set the path to the CMake and CTest executables.
parser.add_argument("--cmake_path", default="cmake", type=Path, help="Path to the CMake program.")
parser.add_argument("--ctest_path", default="ctest", type=Path, help="Path to the CTest program.")
parser.add_argument(
"--cmake_generator",
choices=[
"MinGW Makefiles",
"Ninja",
"NMake Makefiles",
"Unix Makefiles",
"Visual Studio 17 2022",
"Xcode",
],
default=None,
help="Specify the generator that CMake invokes.",
)
parser.add_argument("--ort_home", default=None, type=Path, help="Root directory of onnxruntime.")
parser.add_argument("--use_cuda", action="store_true", help="Whether to use CUDA. Default is to not use cuda.")
parser.add_argument(
"--cuda_home",
type=Path,
help="Path to CUDA home. Read from CUDA_HOME or CUDA_PATH environment variable if not specified."
"Used when --use_cuda is specified.",
)
parser.add_argument("--use_dml", action="store_true", help="Whether to use DML. Default is to not use DML.")
# The following options are mutually exclusive (cross compiling options such as android, ios, etc.)
platform_group = parser.add_mutually_exclusive_group()
platform_group.add_argument("--android", action="store_true", help="Build for Android")
platform_group.add_argument("--ios", action="store_true", help="Build for ios")
# Android options
parser.add_argument(
"--android_abi",
default="arm64-v8a",
choices=["armeabi-v7a", "arm64-v8a", "x86", "x86_64"],
help="Specify the target Android Application Binary Interface (ABI)",
)
parser.add_argument("--android_api", type=int, default=27, help="Android API Level, e.g. 21")
parser.add_argument(
"--android_home", type=Path, default=_path_from_env_var("ANDROID_HOME"), help="Path to the Android SDK."
)
parser.add_argument(
"--android_ndk_path",
type=Path,
default=_path_from_env_var("ANDROID_NDK_HOME"),
help="Path to the Android NDK. Typically `<Android SDK>/ndk/<ndk_version>`.",
)
# iOS build options
parser.add_argument(
"--ios_sysroot",
default="",
help="Specify the location name of the macOS platform SDK to be used",
)
parser.add_argument(
"--ios_toolchain_file",
default="",
help="Path to ios toolchain file, "
"or cmake/genai_ios.toolchain.cmake will be used",
)
parser.add_argument(
"--ios_arch",
type=str,
help="Specify the Target specific architectures for iOS "
"This is only supported on MacOS host",
)
parser.add_argument(
"--ios_deployment_target",
type=str,
help="Specify the minimum version of the target platform "
"This is only supported on MacOS host",
)
# now that all args are added, we can include the full help in the usage message.
parser.usage = f"""
{parser.format_help()}
There are 3 phases which can be individually selected.
The update (--update) phase will run CMake to generate makefiles.
The build (--build) phase will build all projects.
The test (--test) phase will run all unit tests.
Default behavior is --update --build --test for native architecture builds.
Default behavior is --update --build for cross-compiled builds.
If phases are explicitly specified only those phases will be run.
e.g. run with `--build` to rebuild without running the update or test phases
"""
return parser.parse_args()
def _resolve_executable_path(command_or_path: Path, resolution_failure_allowed: bool = False):
"""
Returns the absolute path of an executable.
If `resolution_failure_allowed` is True, returns None if the executable path cannot be found.
"""
executable_path = shutil.which(str(command_or_path))
if executable_path is None:
if resolution_failure_allowed:
return None
else:
raise ValueError(f"Failed to resolve executable path for '{command_or_path}'.")
return Path(executable_path)
def _validate_build_dir(args: argparse.Namespace):
if not args.build_dir:
target_sys = platform.system()
# override if we're cross-compiling
# TODO: Add ios and arm64 support
if args.android:
target_sys = "Android"
elif platform.system() == "Darwin":
# also tweak build directory name for mac builds
target_sys = "macOS"
args.build_dir = Path("build") / target_sys
# set to a config specific build dir. it should exist unless we're creating the cmake setup
is_strict = not args.update
args.build_dir = args.build_dir.resolve(strict=is_strict) / args.config
def _validate_cuda_args(args: argparse.Namespace):
if args.cuda_home:
# default use_cuda to True if cuda_home is specified
args.use_cuda = True
if args.use_cuda:
cuda_home = args.cuda_home if args.cuda_home else _path_from_env_var("CUDA_HOME")
if not cuda_home and util.is_windows():
cuda_home = _path_from_env_var("CUDA_PATH")
cuda_home_valid = cuda_home.exists() if cuda_home else False
if not cuda_home_valid:
raise RuntimeError(
f"cuda_home paths must be specified and valid. cuda_home='{cuda_home}' valid={cuda_home_valid}."
)
args.cuda_home = cuda_home.resolve(strict=True)
def _validate_android_args(args: argparse.Namespace):
if args.android:
if not args.android_home:
raise ValueError("--android_home is required to build for Android")
if not args.android_ndk_path:
raise ValueError("--android_ndk_path is required to build for Android")
args.android_home = args.android_home.resolve(strict=True)
args.android_ndk_path = args.android_ndk_path.resolve(strict=True)
if not args.android_home.is_dir() or not args.android_ndk_path.is_dir():
raise ValueError("Android home and NDK paths must be directories.")
# auto-adjust the cmake generator for cross-compiling Android
original_cmake_generator = args.cmake_generator
if original_cmake_generator not in ["Ninja", "Unix Makefiles"]:
if _resolve_executable_path("ninja", resolution_failure_allowed=True) is not None:
args.cmake_generator = "Ninja"
elif _resolve_executable_path("make", resolution_failure_allowed=True) is not None:
args.cmake_generator = "Unix Makefiles"
else:
raise ValueError(
"Unable to find appropriate CMake generator for cross-compiling for Android. "
"Valid generators are 'Ninja' or 'Unix Makefiles'."
)
if args.cmake_generator != original_cmake_generator:
log.info(f"Setting CMake generator to '{args.cmake_generator}' for cross-compiling for Android.")
# no C# on Android so automatically skip
if not args.skip_csharp:
args.skip_csharp = True
def _validate_ios_args(args: argparse.Namespace):
if args.ios:
if not util.is_mac():
raise ValueError("A Mac host is required to build for iOS")
needed_args = [
args.ios_sysroot,
args.ios_arch,
args.ios_deployment_target,
]
arg_names = [
"--ios_sysroot <the location or name of the macOS platform SDK>",
"--ios_arch <the Target specific architectures for iOS>",
"--ios_deployment_target <the minimum version of the target platform>",
]
have_required_args = all(_ is not None for _ in needed_args)
if not have_required_args:
raise ValueError(
"iOS build on MacOS canceled due to missing arguments: "
+ ", ".join(
val for val, cond in zip(arg_names, needed_args) if not cond
)
)
def _validate_args(args):
# default to all 3 stages
if not args.update and not args.build and not args.test:
args.update = True
args.build = True
args.test = True
# validate args. this updates values in args where applicable (e.g. fully resolve paths).
args.cmake_path = _resolve_executable_path(args.cmake_path)
args.ctest_path = _resolve_executable_path(args.ctest_path)
_validate_build_dir(args)
_validate_cuda_args(args)
_validate_android_args(args)
_validate_ios_args(args)
if args.ort_home:
if not args.ort_home.exists() or not args.ort_home.is_dir():
raise ValueError(f"{args.ort_home} does not exist or is not a directory.")
args.ort_home = args.ort_home.resolve(strict=True)
def _create_env(args: argparse.Namespace):
env = os.environ.copy()
if args.use_cuda:
env["CUDA_HOME"] = str(args.cuda_home)
env["PATH"] = str(args.cuda_home / "bin") + os.pathsep + os.environ["PATH"]
if args.android:
env["ANDROID_HOME"] = str(args.android_home)
env["ANDROID_NDK_HOME"] = str(args.android_ndk_path)
return env
def _get_csharp_properties(args: argparse.Namespace):
# Tests folder does not have a sln file. We use the csproj file to build and test.
# The csproj file requires the platform to be AnyCPU (not "Any CPU")
configuration = f"/p:Configuration={args.config}"
platform = "/p:Platform=Any CPU"
# need an extra config on windows as the actual build output is in the original build dir / config / config
native_lib_path = f"/p:NativeBuildOutputDir={str(args.build_dir / args.config)}"
props = [configuration, platform, native_lib_path]
if args.ort_home:
props.append(f"/p:OrtHome={args.ort_home}")
return props
def update(args: argparse.Namespace, env: dict[str, str]):
"""
Update the cmake build files.
"""
# build the cmake command to create/update the build files
command = [str(args.cmake_path)]
if args.cmake_generator:
command += ["-G", args.cmake_generator]
if util.is_windows():
if not args.cmake_generator:
command += ["-G", "Visual Studio 17 2022", "-A", "x64"]
if args.cmake_generator == "Ninja":
if args.use_cuda:
command += ["-DCUDA_TOOLKIT_ROOT_DIR=" + str(args.cuda_home)]
else:
toolset = "host=x64"
if args.use_cuda:
toolset += ",cuda=" + str(args.cuda_home)
command += ["-T", toolset]
command += [f"-DCMAKE_BUILD_TYPE={args.config}"]
build_wheel = "OFF" if args.skip_wheel else "ON"
command += [
"-S",
str(REPO_ROOT),
"-B",
str(args.build_dir),
"-DCMAKE_POSITION_INDEPENDENT_CODE=ON",
f"-DUSE_CUDA={'ON' if args.use_cuda else 'OFF'}",
f"-DUSE_DML={'ON' if args.use_dml else 'OFF'}",
f"-DENABLE_JAVA={'ON' if args.build_java else 'OFF'}",
f"-DBUILD_WHEEL={build_wheel}",
]
if args.ort_home:
command += [f"-DORT_HOME={args.ort_home}"]
if args.use_cuda:
cuda_arch = 80
cuda_compiler = str(args.cuda_home / "bin" / "nvcc")
command += [f"-DCMAKE_CUDA_COMPILER={cuda_compiler}",
f"-DCMAKE_CUDA_ARCHITECTURES={cuda_arch}"]
if args.android:
command += [
"-DCMAKE_TOOLCHAIN_FILE="
+ str((args.android_ndk_path / "build" / "cmake" / "android.toolchain.cmake").resolve(strict=True)),
f"-DANDROID_PLATFORM=android-{args.android_api}",
f"-DANDROID_ABI={args.android_abi}",
"-DENABLE_PYTHON=OFF",
]
if args.ios:
command += [
"-DCMAKE_SYSTEM_NAME=iOS",
f"-DCMAKE_OSX_SYSROOT={args.ios_sysroot}",
f"-DCMAKE_OSX_ARCHITECTURES={args.ios_arch}",
f"-DCMAKE_OSX_DEPLOYMENT_TARGET={args.ios_deployment_target}",
"-DENABLE_PYTHON=OFF",
"-DCMAKE_TOOLCHAIN_FILE="
+ (
args.ios_toolchain_file
if args.ios_toolchain_file
else "cmake/genai_ios.toolchain.cmake"
),
]
util.run(command, env=env)
def build(args: argparse.Namespace, env: dict[str, str]):
"""
Build the targets.
"""
make_command = [str(args.cmake_path), "--build", str(args.build_dir), "--config", args.config]
if args.parallel:
make_command.append("--parallel")
util.run(make_command, env=env)
if util.is_windows() and not args.skip_csharp:
dotnet = str(_resolve_executable_path("dotnet"))
# Build the library
csharp_build_command = [dotnet, "build", ".",]
csharp_build_command += _get_csharp_properties(args)
util.run(csharp_build_command, cwd=REPO_ROOT / "src" / "csharp")
util.run(csharp_build_command, cwd=REPO_ROOT / "test" / "csharp")
def test(args: argparse.Namespace, env: dict[str, str]):
"""
Run the tests.
"""
ctest_cmd = [str(args.ctest_path), "--build-config", args.config, "--verbose", "--timeout", "10800"]
util.run(ctest_cmd, cwd=str(args.build_dir))
if util.is_windows() and not args.skip_csharp:
dotnet = str(_resolve_executable_path("dotnet"))
csharp_test_command = [dotnet, "test"]
csharp_test_command += _get_csharp_properties(args)
util.run(csharp_test_command, env=env, cwd=str(REPO_ROOT / "test" / "csharp"))
def clean(args: argparse.Namespace, env: dict[str, str]):
"""
Clean the build output.
"""
log.info("Cleaning targets")
cmd_args = [str(args.cmake), "--build", str(args.build_dir), "--config", args.config, "--target", "clean"]
util.run(cmd_args, env=env)
if __name__ == "__main__":
if not (util.is_windows() or util.is_linux() or util.is_mac()):
raise OSError(f"Unsupported platform {sys.platform}.")
arguments = _parse_args()
_validate_args(arguments)
environment = _create_env(arguments)
if arguments.update:
update(arguments, environment)
if arguments.build:
build(arguments, environment)
if arguments.test and not arguments.skip_tests:
test(arguments, environment)