1 This directory contains a number of shell scripts, which we will 2 call the "dev-scripts", that are only used to develop the NDK 3 itself, i.e. they are un-needed when using ndk-build to build 4 applicative native code. 5 6 Their purpose is to handle various sophisticated issues: 7 8 - rebuilding host cross-toolchains for our supported CPU ABIs 9 10 - rebuilding other required host tools (e.g. ndk-stack) from sources 11 12 - rebuilding all target-specific prebuilt binaries from sources 13 (this requires working host cross-toolchains) 14 15 - packaging final NDK release tarballs, including adding samples 16 and documentation which normally live in $NDK/../development/ndk 17 18 This document is here to explain how to use these dev-scripts and how 19 everything is architected / designed, in case you want to maintain it. 20 21 22 I. Organization: 23 ---------------- 24 25 First, a small description of the NDK's overall directory structure: 26 27 $NDK/build/core 28 Contains the main NDK build system used when 'ndk-build'. 29 Relies heavily on GNU Make 3.81+ but isn't used by any of the 30 scripts described here. 31 32 $NDK/build/tools 33 Contains all the dev-scripts that are described in this document. 34 More on this later. 35 36 $NDK/sources/host-tools/ 37 Contains sources of various libraries or programs that will be 38 compiled to generate useful host programs for the final NDK 39 installation. For example, $NDK/sources/host-tools/ndk-stack/ 40 contains the sources of the 'ndk-stack' program. 41 42 $NDK/sources/cxx-stl/ 43 Contains the sources of various C++ runtime and libraries that 44 can be used with 'ndk-build'. See docs/CPLUSPLUS-SUPPORT.html for 45 more details. 46 47 $NDK/sources/cxx-stl/gabi++/ 48 Contains the sources of the GAbi++ C++ runtime library. Note that 49 the dev-script 'build-cxx-stl.sh' can be used to generate prebuilt 50 libraries from these sources, that will be copied under this 51 directory. 52 53 $NDK/sources/cxx-stl/stlport/ 54 Contains the sources of a port of STLport that can be used with ndk-build. 55 The dev-script 'build-cxx-stl.sh' can be used to generate prebuilt 56 libraries from these sources, that will be copied under this directory. 57 58 $NDK/sources/cxx-stl/llvm-libc++/ 59 Contains the sources of a port of LLVM's libc++ that can be used with 60 ndk-build. The dev-script 'build-cxx-stl.sh' can be used to generate 61 prebuilt libraries from these sources, that will be copied under this 62 directory. 63 64 $NDK/sources/cxx-stl/gnu-libstdc++/ 65 This directory doesn't contain sources at all, only an Android.mk. 66 The dev-script 'build-gnu-libstdc++.sh' is used to generate prebuilt 67 libraries from the sources that are located in the toolchain source 68 tree instead. 69 70 $NDK/sources/cxx-stl/system/ 71 This directory contains a few headers used to use the native system 72 Android C++ runtime (with _very_ limited capabilites), a.k.a. 73 /system/lib/libstdc++.so. The prebuilt version of this library is 74 generated by the 'gen-platform.sh' dev-script described later, but 75 it never placed in this directory. 76 77 $NDK/sources/android/libthread_db/ 78 This directory contains the sources of the libthread_db implementation 79 that is linked into the prebuilt target gdbserver binary. Note that two 80 versions are provided, corresponding to different versions of gdb. 81 82 The sources are customized versions of the ones found under 83 $ANDROID/bionic/libthread_db. They contain various changes used to 84 deal with platform bugs. 85 86 $NDK/sources/ 87 The rest of $NDK/sources/ is used to store the sources of helper 88 libraries used with 'ndk-build'. For example, the 'cpu-features' 89 helper library is under $NDK/sources/android/cpu-features/ 90 91 $DEVNDK a.k.a $NDK/../development/ndk 92 This directory contains platform-specific files. The reason why it 93 it is separate from $NDK is because it is not primarly developed in 94 the open. 95 96 More specifically: 97 98 - all $NDK development happens in the public AOSP repository ndk.git 99 100 - any $DEVNDK development that happens in the public AOSP development.git 101 repository is auto-merged to the internal tree maintained by Google 102 103 - $DEVNDK developments that are specific to an yet-unreleased version of 104 the system happen only in the internal tree. They get back-ported to 105 the public tree only when the corresponding system release is 106 open-sourced. 107 108 Having two distinct git repositories makes things easier. Also, you 109 generally don't want to see the churn that goes in the internal tree 110 during development. It's usually not very pretty :) 111 112 $DEVNDK/platforms/android-$PLATFORM, where $PLATFORM is a decimal number 113 Contains all files that are specific to a given API level $PLATFORM, 114 that were not already defined for the previous API level. 115 116 For example, android-3 corresponds to Android 1.5, and android-4 117 corresponds to Android 1.6. The platforms/android-4 directory only 118 contains files that are either new or modified, compared to android-3. 119 120 $DEVNDK/platforms/android-$PLATFORM/include/ 121 Contains all system headers exposed by the NDK for a given platform. 122 All these headers are independent from the CPU architecture of target devices. 123 124 $DEVNDK/platforms/android-$PLATFORM/arch-$ARCH/ 125 Contains all files that are specific to a given $PLATFORM level and a 126 specific CPU architecture. $ARCH is typically 'arm' or 'x86' 127 128 $DEVNDK/platforms/android-$PLATFORM/arch-$ARCH/include/ 129 Contains all the architecture-specific headers for a given API level. 130 131 $DEVNDK/platforms/android-$PLATFORM/arch-$ARCH/lib/ 132 Contains several CPU-specific object files and static libraries that 133 are required to build the host cross-toolchains properly. 134 135 Before NDK r7, this also contains prebuilt system shared libraries that 136 had been hand-picked from various platform builds. These have been 137 replaced by symbol list files instead (see below). 138 139 $DEVNDK/platforms/android-$PLATFORM/arch-$ARCH/symbols/ 140 Contains, for each system shared library exposed by the NDK, two 141 files describing the dynamic symbols it exports, for example, for 142 the C library: 143 144 libc.so.functions.txt -> list of exported function names 145 libc.so.variables.txt -> list of exported variable names 146 147 These files were introduced in NDK r7 and are used to generate "shell" 148 shared libraries that can be used by ndk-build at link time. The shell 149 libraries only provide dynamic symbols to 150 151 These files can be generated from a given platform build using the 152 'dev-platform-import.sh' dev-script, described later in this document. 153 154 This is handy to compare which symbols were added between platform 155 releases (and check that nothing disappeared). 156 157 $DEVNDK/platforms/android-$PLATFORM/samples/ 158 Contains samples that are specific to a given API level. These are 159 usually copied into $INSTALLED_NDK/samples/ by the 'gen-platforms.sh' 160 script. 161 162 163 $NDK/platforms/ 164 Note to be confused with $DEVNDK/platforms/, this directory is not 165 part of the NDK git directory (and is spefically listed in $NDK/.gitignore) 166 but of its final installation. 167 168 Its purpose is to hold the fully expanded platform-specific files. 169 This means that, unlike $DEVNDK/platforms/android-$PLATFORM, the 170 $NDK/platforms/android-$PLATFORM will contain _all_ the files that 171 are specific to API level $PLATFORM. 172 173 Moreover, the directory is organized slightly differently, i.e. as 174 toolchain sysroot, i.e. for each supported $PLATFORM and $ARCH values, 175 it provides two directories: 176 177 $NDK/platforms/android-$PLATFORM/arch-$ARCH/usr/include 178 $NDK/platforms/android-$PLATFORM/arch-$ARCH/usr/lib 179 180 Notice the 'usr' subdirectory here. It is required by GCC to be able 181 to use the directories with --with-sysroot. For example, to generate 182 binaries that target API level 5 for the arm architecture, one would 183 use: 184 185 $TOOLCHAIN_PREFIX-gcc --with-sysroot=$NDK/platforms/android-5/arch-arm 186 187 Where $TOOLCHAIN_PREFIX depends on the exact toolchain being used. 188 189 The dev-script 'gen-platforms.sh' is used to populate $NDK/platforms. 190 Note that by default, the script does more, see its detailed description 191 below. 192 193 Generally, everything dev-script supports the --help option to display a 194 description of the program and the list of all supported options. Also, 195 debug traces can be activated by using the --verbose option. Use it several 196 times to increase the level of verbosity. 197 198 Note that all Windows host programs can be built on Linux if you have the 199 'mingw32' cross-toolchain installed ('apt-get install mingw32' on Debian or 200 Ubuntu). You will need to add the '--mingw' option when invoking the script. 201 202 All dev-scripts rebuilding host programs on Linux and Darwin will only 203 generate 32-bit programs by default. You can experiment with 64-bit binary 204 generation by adding the '--try-64' option. Note that as of now, 64-bit 205 binaries are never distributed as part of official NDK releases. 206 207 When building 32-bit Linux host programs, the dev-scripts will look for 208 $NDK/../prebuilts/gcc/linux-x86/host/i686-linux-glibc2.7-4.6/, which is 209 part of the Android platform source tree. It is a special toolchain that 210 ensures that the generated programs can run on old systems like Ubuntu 8.04 211 that only have GLibc 2.7. Otherwise, the corresponding binaries may not run 212 due to ABI changes in mor recent versions of GLibc. 213 214 This shall always be used to official NDK Linux releases. 215 216 217 II. Host toolchains: 218 -------------------- 219 220 The host toolchains are the compiler, linker, debugger and other crucial 221 programs used to generate machine code for the target Android 222 system supported by the NDK. 223 224 225 II.1 Getting the toolchain sources: 226 - - - - - - - - - - - - - - - - - - 227 228 The NDK toolchain sources are located under the toolchain/ directly 229 from the top of the public AOSP repositories. There are actually several 230 git repositories of interest here: 231 232 binutils/ 233 build/ 234 gcc/ 235 gdb/ 236 gold/ 237 gmp/ 238 mpfr/ 239 240 The AOSP toolchain/ repository contains sources corresponding to toolchains 241 that are used to build the Android platform tree. For various reasons, they 242 need to be slightly patched for use with the NDK. 243 244 All patches are located under $NDK/build/tools/toolchain-patches/ 245 246 The script 'download-toolchain-sources.sh' can be used to download and patch 247 the toolchain sources from the server. You simply need to give it the name of 248 a destination directory, for example: 249 250 $NDK/build/tools/download-toolchain-sources.sh /tmp/ndk-$USER/src 251 252 Note that this process can take several minutes. If you happen to already 253 have a clone of the repo toolchain/ tree, you can speed this up considerably 254 using the --git-reference or --git-base option. See --help for more options 255 and details. 256 257 Note, to create a local toolchain repo: 258 259 mkdir -p /tmp/ndk-$USER/toolchain 260 cd /tmp/ndk-$USER/toolchain 261 repo init https://android.googlesource.com/toolchain/manifest.git 262 repo sync 263 264 then use with: 265 266 $NDK/build/tools/download-toolchain-sources.sh \ 267 --git-reference=/tmp/ndk-$USER/toolchain \ 268 /tmp/ndk-$USER/src 269 270 271 The script doesn't simply copy the current HEAD of the toolchain 272 repositories. Instead, it will checkout the sources as they were 273 at a given known date, defined as TOOLCHAIN_GIT_DATE in 'prebuilt-common.sh'. 274 275 If you adjust TOOLCHAIN_GIT_DATE, verify that all patches still apply 276 correctly, you might need to remove patches if they have been integrated 277 into the toolchain/ repositories. 278 279 Note: you can avoid the patching with the --no-patches option to 280 'download-toolchain-patches.sh', and later use 'patch-sources.sh' manually 281 to verify that the patches apply cleanly. 282 283 The toolchains binaries are typically placed under the directory 284 $NDK/toolchains/$NAME/prebuilt/$SYSTEM, where $NAME is the toolchain name's 285 full name (e.g. arm-linux-androideabi-4.6), and $SYSTEM is the name of the 286 host system it is meant to run on (e.g. 'linux-x86', 'windows' or 'darwin-x86') 287 288 289 I.2. Building the toolchains: 290 - - - - - - - - - - - - - - - 291 292 After downloading and patching the toolchain sources, you will need to build 293 a proper "sysroot" directory before being able to configure/build them. 294 295 A sysroot is a directory containing system headers and libraries that the 296 compiler will use to build a few required target-specific binaries (e.g. libgcc.a) 297 298 To do that, use: 299 300 $NDK/build/tools/gen-platforms.sh --minimal 301 302 This will populate $NDK/platforms/ with just the files necessary to rebuild the 303 toolchains. Note that without the --minimal option, the script will fail without 304 prebuilt toolchain binaries. 305 306 Once the sysroots are in place, use 'build-gcc.sh' by providing the path 307 to the toolchain sources root directory, a destination NDK installation 308 directory to build, and the full toolchain name. 309 310 For example, to rebuild the arm and x86 prebuilt toolchain binaries in the 311 current NDK directory (which can be handy if you want to later use them to 312 rebuild other target prebuilts or run tests), do: 313 314 $NDK/build/tools/build-gcc.sh /tmp/ndk-$USER/src $NDK arm-linux-androideabi-4.6 315 $NDK/build/tools/build-gcc.sh /tmp/ndk-$USER/src $NDK x86-4.6 316 317 Here, we assume you downloaded the toolchain sources in /tmp/ndk-$USER/src 318 as described in the previous section. 319 320 This operation can take some time. The script automatically performs a parallel 321 build to speed up the build on multi-core machine (use the -j<number> option to 322 control this), but the GCC sources are very large, so expect to wait a few 323 minutes. 324 325 For the record, on a 2.4 GHz Xeon with 16 Hyper-threaded cores and 12GB of 326 memory, rebuilding each toolchain takes between 2 and 4 minutes. 327 328 You need to be on Linux to build the Windows binaries, using the "mingw32" 329 cross-toolchain (install it with "apt-get install mingw32" on Ubuntu). To do so 330 use the "--mingw" option, as in: 331 332 $NDK/build/tools/build-gcc.sh --mingw \ 333 /tmp/ndk-$USER/src $NDK arm-linux-androideabi-4.6 334 335 $NDK/build/tools/build-gcc.sh --mingw \ 336 /tmp/ndk-$USER/src $NDK x86-4.6 337 338 The corresponding binaries are installed under $NDK/toolchains/$NAME/prebuilt/windows 339 Note that these are native Windows programs, not Cygwin ones. 340 341 Building the Windows toolchains under MSys and Cygwin is completely unsupported 342 and highly un-recommended: even if it works, it will probably take several 343 hours, even on a powerful machine :-( 344 345 The Darwin binaries must be generated on a Darwin machine. Note that the script 346 will try to use the 10.5 XCode SDK if it is installed on your system. This 347 ensures that the generated binaries run on Leopard, even if you're building 348 on a more recent version of the system. 349 350 Once you've completed your builds, you should be able to generate the other 351 target-specific prebuilts. 352 353 354 III. Target-specific prebuilt binaries: 355 --------------------------------------- 356 357 A final NDK installation comes with a lot of various target-specific prebuilt 358 binaries that must be generated from sources once you have working host toolchains. 359 360 III.1.: Preparation of platform sysroots: 361 - - - - - - - - - - - - - - - - - - - - - 362 363 Each target prebuilt is handled by a specific dev-script. HOWEVER, all these script 364 require that you generate a fully populated $NDK/platforms/ directory first. To do 365 that, simply run: 366 367 $NDK/gen-platforms.sh 368 369 Note that we used this script with the --minimal option to generate the host 370 toolchains. That's because without this flag, the script will also auto-generate 371 tiny versions of the system shared libraries that will be used at link-time when 372 building our target prebuilts. 373 374 III.2.: Generation of gdbserver: 375 - - - - - - - - - - - - - - - - - 376 377 A target-specific 'gdbserver' binary is required. This is a small program 378 that is run on the device through 'ndk-gdb' during debugging. For a variety 379 of technical reasons, it must be copied into a debuggable project's output 380 directory when 'ndk-build' is called. 381 382 The prebuilt binary is placed under $NDK/toolchains/$NAME/prebuilt/gdbserver 383 in the final NDK installation. You can generate with 'build-gdbserver.sh' and 384 takes the same parameters than 'build-gcc.sh'. So one can do: 385 386 $NDK/build/tools/build-gcc.sh /tmp/ndk-$USER/src $NDK arm-linux-androideabi-4.6 387 $NDK/build/tools/build-gcc.sh /tmp/ndk-$USER/src $NDK x86-4.6 388 389 390 III.3. Generating C++ runtime prebuilt binaries: 391 - - - - - - - - - - - - - - - - - - - - - - - - 392 393 Sources and support files for several C++ runtimes / standard libraries are 394 provided under $NDK/sources/cxx-stl/. Several dev-scripts are provided to 395 rebuild their binaries. The scripts place them to their respective location 396 (e.g. the GAbi++ binaries will go to $NDK/sources/cxx-stl/gabi++/libs/) 397 unless you use the --out-dir=<path> option. 398 399 Note that: 400 401 - each script will generate the binaries for all the CPU ABIs supported by 402 the NDK, e.g. armeabi, armeabi-v7a, x86 and mips. You can restrict them using 403 the --abis=<list> option though. 404 405 - the GNU libstdc++ dev-script requires the path to the toolchain sources, 406 since this is where the library's sources are located. 407 408 An example usage would be: 409 410 $NDK/build/tools/build-cxx-stl.sh --stl=gabi++ 411 $NDK/build/tools/build-cxx-stl.sh --stl=stlport 412 $NDK/build/tools/build-cxx-stl.sh --stl=libc++ 413 $NDK/build/tools/build-gnu-libstdc++.sh /tmp/ndk-$USER/src 414 415 Note that generating the STLport and GNU libstdc++ binaries can take a 416 few minutes. You can follow the build by using the --verbose option to display 417 what's going on. 418 419 420 IV. Other host prebuilt binaries: 421 --------------------------------- 422 423 There are a few other host prebuilt binaries that are needed for a full 424 NDK installation. Their sources are typically installed under 425 $NDK/sources/host-tools/ 426 427 Note that the corresponding dev-script recognize the --mingw and --try-64 428 options described at the end of section I above. 429 430 431 IV.1.: Building 'ndk-stack': 432 - - - - - - - - - - - - - - 433 434 The 'build-ndk-stack.sh' script can be used to rebuild the 'ndk-stack' 435 helper host program. See docs/NDK-STACK.html for a usage description. 436 To build it, just do: 437 438 $NDK/build/tools/build-ndk-stack.sh 439 440 IV.2.: Building 'ndk-depends': 441 - - - - - - - - - - - - - - - 442 443 Similar to 'ndk-stack', see the 'build-ndk-depends.sh' script. 444 445 446 V. Packaging all prebuilts: 447 --------------------------- 448 449 Generating all the prebuilt binaries takes a lot of time and is no fun. 450 To avoid doing it again and again, it is useful to place all the generated 451 files aside in special tarballs. 452 453 Most dev-scripts generating them typically support a --package-dir=<path> 454 option to do this, where <path> points to a directory that will store 455 compressed tarballs of the generated binaries. 456 457 For example, to build and package the GAbi++ binaries, use: 458 459 $NDK/build/tools/build-cxx-stl.sh --stl=gabi++ --package-dir=/tmp/ndk-$USER/prebuilt/ 460 461 In NDK r7, this will actually create three tarballs (one per supported ABI), 462 under the directory /tmp/ndk-$USER/prebuilt/, i.e.: 463 464 gabixx-libs-armeabi.tar.bz2 465 gabixx-libs-armeabi-v7a.tar.bz2 466 gabixx-libs-x86.tar.bz2 467 ... 468 469 Note that these tarballs are built to be uncompressed from the top-level 470 of an existing NDK install tree. 471 472 Similarly, to rebuild the STLport binaries and package them: 473 474 $NDK/build/tools/build-cxx-stl.sh --stl=stlport --package-dir=/tmp/ndk-$USER/prebuilt 475 476 A dev-script is provided to rebuild _and_ package all prebuilts. It is called 477 'rebuild-all-prebuilt.sh'. Note that by default, it will automatically 478 invoke 'download-toolchain-sources.sh', and place the prebuilt tarballs 479 under /tmp/ndk-$USER/prebuilt-$DATE, where $DATE is the current date in 480 ISO format (e.g. 20110915 for the 15th of September of 2011). 481 482 If you have already downloaded the toolchain sources, use the 483 --toolchain-src-dir=<path> option to save a few minutes to your rebuild, 484 as in: 485 486 $NDK/build/tools/rebuild-all-prebuilt.sh \ 487 --toolchain-src-dir=/tmp/ndk-$USER/src 488 489 By default, this only rebuilds the host prebuilds for the current host 490 system. You can use --mingw to force the generation of Windows binaries on 491 Linux. 492 493 Additionally, you can use the --darwin-ssh=<hostname> option to launch the 494 build of the Darwin binaries from a Linux machine, by using ssh to access a 495 remote Darwin machine. The script will package all required sources into a 496 temporary tarball, copy it to the remote machine, launch the build there, 497 then copy back all binaries to your own machine. 498 499 This means that it is possible to generate the host binaries for all supported 500 host systems from Linux (provided you have ssh access to a Darwin machine). 501 502 Alternatively, you can run 'rebuild-all-prebuilt.sh' on a Darwin machine. 503 504 Once you have used the script three times (once per supported host systems), 505 you should have plenty of files under /tmp/ndk-$USER/prebuilt-$DATE. 506 For the record, with NDK r7, the list was: 507 508 509 VI. Packaging NDK releases: 510 --------------------------- 511 512 Use the 'package-release.sh' dev-script to generate full NDK release packages. 513 These contain everything needed by a typical NDK user, including: 514 515 - all prebuilt binaries (host toolchains, host tools, target libs, etc...) 516 - all samples (including those collected from $DEVNDK/platforms/) 517 - all documentation 518 519 You need to have a directory containing prebuilt tarballs, as described in 520 the previous section. You can use it as: 521 522 $NDK/build/tools/package-release.sh \ 523 --release=<name> \ 524 --systems=<list> \ 525 --arch=<list> \ 526 --prebuilt-dir=<path> 527 528 The --release option is optional and allows you to provide a name for your 529 generated NDK archive. More specifically, the archive file name will be 530 something like android-ndk-$RELEASE-$SYSTEM.tar.bz2, where $RELEASE is 531 the release name, and $SYSTEM the supported host system (e.g. linux-x86). 532 533 By default, i.e. without the option, $RELEASE will be set to the current $DATE. 534 535 The --systems=<list> is optional, but can be used to limit the number of host 536 systems you want to generate for. <list> must be a comma-separated list of 537 system names (from 'linux-x86', 'windows' and 'darwin-x86'). This is useful 538 if you're working on a experimental feature and don't have the time to 539 regenerate the host toolchains for all systems. It allows you to generate 540 an experimental package that you can distribute to third-party for 541 experimentation. 542 543 By default, i.e. without the option, the scripts tries to build NDK archives 544 for all supported host systems. 545 546 The --arch=<list> is also optional, but can be used ot limit the number of 547 target architectures you want to generate for. <list> must be a comma-separated 548 list of CPU architectures (e.g. from 'arm' and 'x86'). Without the option, 549 this will try to build packages that support all architectures. 550 551 Finally, --prebuilt-dir=<path> must point to the directory that contains the 552 prebuilt tarballs described in section V. Following our previous example, one 553 could use --prebuilt-dir=/tmp/ndk-$USER/prebuilt here. 554 555 556 VI. Testing: 557 ------------ 558 559 The $NDK/tests directory contains a number of NDK unit-tests that can be 560 used to verify that the generated NDK packages or the working NDK tree still 561 behave correctly. 562 563 If you have an NDK package archive, you can run the following to run the 564 test suite against it: 565 566 $NDK/tests/run-tests.sh --package=<ndk-archive> 567 568 This will uncompress the NDK archive in a temporary directory, then run all 569 the tests with it. When all tests have run, the temporary directory is removed 570 automatically. 571 572 You can also point to an existing NDK installation with --ndk=<path>, as in: 573 574 $NDK/tests/run-tests.sh --ndk=<path> 575 576 Where <path> points to another NDK installation. The script will run the 577 test suite present under $NDK/tests/, not the one in the remote NDK directory. 578 579 If you don't use any option, the test suite will be run with the current NDK 580 directory. This can only work if you have generated or unpacked all prebuilt 581 archives into it before that. 582 583 You can get more traces from the tests by using --verbose. Use it twice to see 584 even more traces. 585 586 There are several kinds of tests: 587 588 - 'build tests' are used to test the building capabilities of the NDK. 589 I.e. the tests will only use them to check that the NDK build system 590 didn't regress. The corresponding generated binaries are never used 591 otherwise. 592 593 - 'device tests' are used to test both the build and the behaviour of 594 the generated code. If the 'adb' program is in your path, and have 595 one device or emulator connected to your host machine, 'run-tests.sh' 596 will automatically upload, run and cleanup these tests for you. 597 598 If adb is not in your path, or no device is connected, run-tests.sh 599 will simply print a warning and carry on. 600 601 602 Whenever you add a feature to the NDK, or fix a bug, it is recommended to 603 add a unit test to check the feature or the fix. Use $NDK/tests/build for 604 build tests, and $NDK/tests/device for device tests. 605 606