1Android NDK Overview 2 3Introduction: 4 5The Android NDK is a set of tools that allows Android application developers 6to embed native machine code compiled from C and/or C++ source files into 7their application packages. 8 9IMPORTANT: 10 The Android NDK can only be used to target Android system images 11 running Cupcake (a.k.a 1.5) or later versions of the platform. 12 13 1.0 and 1.1 system images are specifically *not* supported due to 14 subtle ABI and toolchain changes that happened for the 1.5 release. 15 16 17I. Android NDK Goals: 18--------------------- 19 20The Android VM allows your application's source code to call methods 21implemented in native code through the JNI. In a nutshell, this means that: 22 23 - Your application's source code will declare one or more methods 24 with the 'native' keyword to indicate that they are implemented through 25 native code. E.g.: 26 27 native byte[] loadFile(String filePath); 28 29 - You must provide a native shared library that contains the 30 implementation of these methods, which will be packaged into your 31 application's .apk. This library must be named according to standard 32 Unix conventions as lib<something>.so, and shall contain a standard JNI 33 entry point (more on this later). For example: 34 35 libFileLoader.so 36 37 - Your application must explicitely load the library. For example, to load 38 it at application startup, simply add the following to its source code: 39 40 static { 41 System.loadLibrary("FileLoader"); 42 } 43 44 Note that you should not use the 'lib' prefix and '.so' suffix here. 45 46 47The Android NDK is a complement to the Android SDK that helps you to: 48 49 - Generate JNI-compatible shared libraries that can run on the Android 50 1.5 platform (and later) running on ARM CPUs. 51 52 - Copy the generated shared libraries to a proper location of your 53 application project path, so they will be automatically added to your 54 final (and signed) .apks 55 56 - In later revisions of the NDK, we intend to provide tools that help 57 debug your native code through a remote gdb connection and as much 58 source/symbol information as possible. 59 60Moreover, the Android NDK provides: 61 62 - A set of cross-toolchains (compilers, linkers, etc..) that can 63 generate native ARM binaries on Linux, OS X and Windows (with Cygwin) 64 65 - A set of system headers corresponding to the list of stable native APIs 66 supported by the Android platform. This corresponds to definitions that 67 are guaranteed to be supported in all later releases of the platform. 68 69 They are documented in the file docs/STABLE-APIS.TXT 70 71 IMPORTANT: 72 Keep in mind that most of the native system libraries in Android system 73 images are not frozen and might changed drastically, or even deleted, 74 in later updates and releases of the platform. 75 76 - A build system that allow developers to only write very short build files 77 to describe which sources need to be compiled, and how. The build system 78 deals with all the hairy toolchain/platform/CPU/ABI specifics. Moreover, 79 later updates of the NDK can add support for more toolchains, platforms, 80 system interfaces without requiring changes in the developer's build 81 files (more on this later). 82 83 84II. Android NDK Non-Goals: 85-------------------------- 86 87The NDK is *not* a good way to write generic native code that runs on Android 88devices. In particular, your applications should still be written in the Java 89programming language, handle Android system events appropriately to avoid the 90"Application Not Responding" dialog or deal with the Android application 91life-cycle. 92 93Note however that is is possible to write a sophisticated application in 94native code with a small "application wrapper" used to start/stop it 95appropriately. 96 97A good understanding of JNI is highly recommended, since many operations 98in this environment require specific actions from the developers, that are 99not necessarily common in typical native code. These include: 100 101 - Not being able to directly access the content of VM objects through 102 direct native pointers. E.g. you cannot safely get a pointer to a 103 String object's 16-bit char array to iterate over it in a loop. 104 105 - Requiring explicit reference management when the native code wants to 106 keep handles to VM objects between JNI calls. 107 108 109The NDK only provides system headers for a very limited set of native 110APIs and libraries supported by the Android platform. While a typical 111Android system image includes many native shared libraries, these should 112be considered an implementation detail that might change drastically between 113updates and releases of the platform. 114 115If an Android system library is not explicitely supported by the NDK 116headers, then applications should not depend on it being available, or 117they risk breaking after the next over-the-air system update on various 118devices. 119 120Selected system libraries will gradually be added to the set of stable NDK 121APIs. 122 123 124III. NDK development in practice: 125--------------------------------- 126 127Here's a very rough overview of how you can develop native code with the 128Android NDK: 129 130 1/ Run build/host-setup.sh to configure the NDK 131 132 2/ Place your native sources under $PROJECT/jni/... 133 134 3/ Write $PROJECT/jni/Android.mk to describe your sources 135 to the NDK build system 136 137 4/ Write apps/<myapp>/Application.mk to describe your application 138 and the native sources it needs to the NDK build system 139 140 5/ Build your native code by running "make APP=<myapp>" 141 in the top-level NDK directory. 142 143The last step will copy, in case of success, the stripped shared libraries 144your application needs to your application's root project directory. You 145will then need to generate your final .apk through the usual means. 146 147Now, for a few more details: 148 149 150III.1/ Configuring the NDK: 151- - - - - - - - - - - - - - 152 153After installing the NDK as described in docs/INSTALL.TXT, you should call 154the 'build/host-setup.sh' script to configure your NDK. 155 156This script is used to probe your host system and verify a few pre-requisites. 157It will then generate a configuration file (e.g. out/host/config-host.mk) that 158is later used during NDK builds. 159 160In some cases, this might instruct you to download an archive containing 161prebuilt toolchain binaries for your development platform, the unzip it 162to the NDK root directory. The message should contain enough information 163to let you do that. 164 165If you forget this step, trying to build with the NDK will generate an 166error message telling you what to do. 167 168 169III.2/ Placing C and C++ sources: 170- - - - - - - - - - - - - - - - - 171 172You should place your native sources under the following directory: 173 174 $PROJECT/jni/ 175 176Where $PROJECT corresponds to the path of your Android application 177project. 178 179You are pretty free to organize the content of 'jni' as you want, 180the directory names and structure here will not influence the final 181generated application packages, so you don't have to use pseudo-unique 182names like com.<mycompany>.<myproject> as is the case for application 183package names. 184 185Note that C and C++ sources are supported. The default C++ file extensions 186supported by the NDK is '.cpp', but other extensions can be handled as well 187(see docs/ANDROID-MK.TXT for details). 188 189It is possible to store your sources in a different location by adjusting 190your Android.mk file (see below). 191 192 193III.3/ Writing an Android.mk build script: 194- - - - - - - - - - - - - - - - - - - - - - 195 196An Android.mk file is a small build script that you write to describe your 197sources to the NDK build system. Its syntax is described in details in 198the file docs/ANDROID-MK.TXT. 199 200In a nutshell, the NDK groups your sources into "modules", where each module 201can be one of the following: 202 203 - a static library 204 - a shared library 205 206You can define several modules in a single Android.mk, or you can write 207several Android.mk files, each one defining a single module. 208 209Note that a single Android.mk might be parsed several times by the build 210system so don't assume that certain variables are not defined in them. 211By default, the NDK will look for the following build script: 212 213 $PROJECT/jni/Android.mk 214 215If you want to define Android.mk files in sub-directories, you should 216include them explicitely in your top-level Android.mk. There is even 217a helper function to do that, i.e. use: 218 219 include $(call all-subdir-makefiles) 220 221This will include all Android.mk files in sub-directories of the current 222build file's path. 223 224 225III.4/ Writing an Application.mk build file: 226- - - - - - - - - - - - - - - - - - - - - - - 227 228While an Android.mk file describes your modules to the build system, you 229need to write an Application.mk file to describe your application and the 230modules it requires. This file must be located in: 231 232 $NDK/apps/<myapp>/Application.mk 233 234Where <myapp> is a short descriptive name for your application that will 235be used to invoke the NDK build (and not go into final APKs). The file is 236used to provide the following to the NDK build: 237 238 - The location of your Android application's project path 239 240 - The list of NDK modules that is required by your application. 241 This should really be a list of 'shared library' modules. 242 243 - Optional information, like whether you want a release or debug 244 build, specific C or C++ compiler flags and others. 245 246 - Planned: the list of specific platforms/CPUs you want to explicitely 247 target (currently only one is supported). 248 249The syntax of an Application.mk file is very simple and is described in 250docs/APPLICATION-MK.TXT 251 252You can define several Application.mk corresponding to different builds 253of the same application, for example: 254 255 $NDK/apps/release/Application.mk 256 $NDK/apps/debug/Application.mk 257 258 259III.5/ Invoke the NDK build system: 260- - - - - - - - - - - - - - - - - - 261 262On the command-line, go to the top-level NDK directory, then invoke the 263build system with: 264 265 make APP=<myapp> 266 267Where 'make' refers to GNU Make, and <myapp> is the name of one of the 268subdirectories of '$NDK/apps/' 269 270This will try to build all modules with relevant options, the final 271shared libraries listed by your Application.mk and, in case of success, 272will copy stripped versions of the shared libraries to your application's 273project path. (Note that unstripped versions are kept for debugging 274purposes, there is no need to copy unstripped binaries to a device). 275 276 277 278IV. Debugging support: 279- - - - - - - - - - - - 280 281Debugging your native code with this initial release of the NDK is still 282very rough. 283 284Note that we plan to make this much easier in a later NDK release, all of 285this without changing your sources, Android.mk and Application.mk files. 286