From e4a19212b222da57a749deaab962e9f7f47c1b2c Mon Sep 17 00:00:00 2001 From: Richard Diamond Date: Thu, 9 Jul 2015 13:33:15 -0500 Subject: [PATCH] Add notes for how to build a cross compiler for a few platforms. Android && iOS's instructions were retreived from the old wiki. --- src/doc/trpl/SUMMARY.md | 1 + src/doc/trpl/cross-compilers.md | 421 ++++++++++++++++++++++++++++++++ 2 files changed, 422 insertions(+) create mode 100644 src/doc/trpl/cross-compilers.md diff --git a/src/doc/trpl/SUMMARY.md b/src/doc/trpl/SUMMARY.md index 24686e772e3c3..589e314763d69 100644 --- a/src/doc/trpl/SUMMARY.md +++ b/src/doc/trpl/SUMMARY.md @@ -20,6 +20,7 @@ * [FFI](ffi.md) * [Borrow and AsRef](borrow-and-asref.md) * [Release Channels](release-channels.md) + * [Cross Compilers](cross-compilers.md) * [Syntax and Semantics](syntax-and-semantics.md) * [Variable Bindings](variable-bindings.md) * [Functions](functions.md) diff --git a/src/doc/trpl/cross-compilers.md b/src/doc/trpl/cross-compilers.md new file mode 100644 index 0000000000000..310b24345fff0 --- /dev/null +++ b/src/doc/trpl/cross-compilers.md @@ -0,0 +1,421 @@ +% Cross Compilers + +Rust is designed to be run anywhere. Anywhere! To do this, however, you're have +to build a cross compiler for the desired target yourself: the binary distributions +don't come equipped with this ability. + +## General + +In general, to create a cross compiler one has to add `--target=` to `configure`. However a few platforms use special toolchains for +linking or have other idiosyncrasies and thus require more involved steps to +create. If you write a port, and want to incorporate the port in Rust proper, +please add a section on this page! :) + +## PNaCl/JS + +To build a PNaCl/JS cross compiler, you'll need to first abtain the +`pepper_canary` toolchain from the [Native Client SDK][nacl-sdk]. Then configure +Rust, where `NACL_SDK_ROOT` points to where `pepper_canary` is installed, with: + + $ CC="$NACL_SDK_ROOT/toolchain/$($NACL_SDK_ROOT/tools/getos.py)_pnacl/bin/clang" + CXX="$NACL_SDK_ROOT/toolchain/$($NACL_SDK_ROOT/tools/getos.py)_pnacl/bin/clang++" + /path/to/rust/configure --target=le32-unknown-nacl --nacl-cross-path=$NACL_SDK_ROOT + --llvm-root=$NACL_SDK_ROOT/toolchain/$($NACL_SDK_ROOT/tools/getos.py)_pnacl + --enable-libcpp --enable-clang + +Then build: `make -j $(nproc)`. After the build completes, install! + +To use Cargo with your cross compiler, you'll first need to set up a few env vars: + + $ export PNACL_TOOLCHAIN=$NACL_SDK_ROOT/toolchain/$($NACL_SDK_ROOT/tools/getos.py)_pnacl; + export CC_le32_unknown_nacl=$($NACL_SDK_ROOT/tools/nacl_config.py -t pnacl --tool=cc); + export CXX_le32_unknown_nacl=$($NACL_SDK_ROOT/tools/nacl_config.py -t pnacl --tool=cxx); + export AR_le32_unknown_nacl=$($NACL_SDK_ROOT/tools/nacl_config.py -t pnacl --tool=ar); + export LD_LIBRARY_PATH=$PNACL_TOOLCHAIN/lib:$LD_LIBRARY_PATH; + export PATH=$PNACL_TOOLCHAIN/bin:$PATH; + +Then you can build with Cargo like: `cargo build --target=le32-unknown-nacl` + +Once you have a `pexe` from Cargo, you can do one of three things: + +1. Produce a stable `pexe`; +2. Translate the bitcode into a Native Client module (ie for debugging); or +3. Send the bitcode to `emcc` (append `bc` to the filename first!) for + Javascript codegening. + +If you are going to use the either 1 or 2, there are some additional resources +available: +* [`pexe-runner`][pexe-runner]: Run non-PPAPI PNaCl modules like one would any other native + binary (on Linux you can also install a binfmt handler). +* [`rust-ppapi`][rust-ppapi]: Rust bindings to parts C Pepper API. + + +[nacl-sdk]: https://developer.chrome.com/native-client/sdk/download + +[pexe-runner]: https://github.com/DiamondLovesYou/pexe-runner + +[rust-ppapi]: https://github.com/DiamondLovesYou/rust-ppapi + +## Android + +Rust has been ported to Android OS running on ARM architecture. Android on other architectures is not yet supported. + +**Important**: Right now, this only works correctly on linux. It can be made to work on OSX, but if you're on OSX, make things easy by setting up a VM. It has been successfully tested on ubuntu 14.04.2 in virtualbox, if you want something super safe, but most any debian-based thing will almost certainly work. + +In the issue tracker, [Android-related issues](https://github.com/rust-lang/rust/issues?labels=A-android) are labelled `A-android`, and [ARM-related issues](https://github.com/rust-lang/rust/issues?labels=A-ARM) are labelled `A-ARM`. + +Instructions to build a cross compiler to Android on ARM, to cross compile Rust sources, and to run resulting binaries follow. + +1. Setup Android NDK standalone tool chain with your specific [API Level](http://developer.android.com/guide/topics/manifest/uses-sdk-element.html#ApiLevels) + + Download android NDK version _r9b_ or later (you might not be able to build jemalloc with previous versions) from http://developer.android.com/tools/sdk/ndk/index.html. for example: + + $ wget http://dl.google.com/android/ndk/android-ndk-r10d-linux-x86_64.bin + $ chmod +x android-ndk-r10d-linux-x86_64.bin + $ ./android-ndk-r10d-linux-x86_64.bin + $ ls android-ndk-r10d -d + android-ndk-r10d + + example command to setup standalone tool chain: + + ~/android-ndk-/build/tools/make-standalone-toolchain.sh \ + --platform=android-14 \ + --arch=arm \ + --install-dir=~/ndk-standalone-14-arm + + `--platform` indicates the Android API level - has successfully been tested with 18 as well + `--install-dir` indicates the directory where the toolchain will be installed. You can pick + more or less anything you feel like. + `--arch` must be set to `arm`, or it won't know what toolchain to use. (Imagine that!) + `--toolchain` can also be set to one of the names of the directories in + `android-ndk-/toolchains/` - keeping in mind it must be one of the arm ones. + Only do this if you know why you want to - otherwise, the default of ...-4.8 will work fine. + + If you're on 64bit linux, the android ndk needs 32bit linux libraries, such as `/lib32/libc.so.6`, `/usr/lib32/libz.so.1`, and `/usr/lib32/libstdc++.so.6.0.17`, so make sure they're installed. + + If you're using ubuntu 64, this command will do: + + apt-get install libc6-i386 lib32z1 lib32stdc++6 + + Some of the tool chain will need to be placed in your path, if you plan to build with cargo. + The easy way to do this is to add these environment variables in ~/.bashrc: + + export ANDROID_NDK=".../path/to/android-ndk-" + export ANDROID_TOOLCHAIN=".../path/to/standalone/toolchain" + export PATH="$PATH:$ANDROID_TOOLCHAIN/bin" + + For example: + + export ANDROID_NDK="$HOME/android-ndk-r10d" + export ANDROID_TOOLCHAIN="$HOME/ndk-standalone-14-arm" + export PATH="$PATH:$ANDROID_TOOLCHAIN/bin" + + Then do `source ~/.bashrc` to update your environment. + +2. Download rustc from git repository + + git clone https://github.com/rust-lang/rust.git + +3. Configure with ndk toolchain path + + mkdir build + cd build + + ../configure \ + --host=x86_64-unknown-linux-gnu \ + --target=arm-linux-androideabi \ + --android-cross-path="$ANDROID_TOOLCHAIN" + +4. Build + + make + + make install + + `make install` will copy rustc binaries and libraries into /usr/local (or as defined with --prefix). You will probably need to do it as root, depending on whether you set --prefix when doing configure. + +5. How to cross compile + +#### With rustc: + + rustc --target=arm-linux-androideabi -C linker=$ANDROID_TOOLCHAIN/bin/arm-linux-androideabi-gcc -C link-args=-pie hello.rs + +`-C link-args=-pie` is a workaround for [issue #17437](https://github.com/rust-lang/rust/issues/17437). + +#### With cargo: + +Because of the need for the `-C link-args=-pie` to rustc, a horrifying hack is required +to build code for android using cargo: putting a `rustc` script earlier on PATH, that will +add the `-C link-args=-pie` argument. This is because cargo doesn't allow us to directly +set the link-args argument. This script will set up the workaround automatically: + +```sh +if [[ ! -d ~/bin ]]; then + mkdir ~/bin +fi +# check if ~/bin comes before the location of rustc +# and if it doesn't, put it there +if [[ $PATH != *"$HOME/bin:"*"$(dirname $(which rustc))" ]]; then + echo 'export PATH="$HOME/bin:$PATH"' >> ~/.bashrc +fi + +cat > ~/bin/rustc << +#!/bin/bash +exec "$(which rustc)" -C link-args=-pie "\$@" +END +chmod +x ~/bin/rustc +``` + +After running that script, make sure to `source ~/.bashrc` again, and then the +following instructions should work. + + 1. Create a .cargo directory in your project's root dir + 2. Create a config file inside the .cargo directory (proj/.cargo/config) + 3. Place your deps/linker information inside that file. for example: + + [target.arm-linux-androideabi] + linker = "arm-linux-androideabi-gcc" + + 4. Then, to compile: + + cargo build --target=arm-linux-androideabi + + 5. Copy `target/arm-linux-androideabi/` to your device, *not* `target/`. (This may change depending on how some cargo issues about the paths being confusing get resolved.) + +6. How to run on Android + + 1. If necessary, get your binary off your VM or set up USB to pass through. How to do this varies a lot; if you copy the binary off your vm, you'll need the android sdk on your host, and adb set up in your host's PATH. + + 2. Push your binary + + $ adb push hello /data/local/tmp + + 3. Run using adb shell + + $ adb shell + shell@hammerhead:/ $ cd /data/local/tmp + shell@hammerhead:/ $ chmod 755 hello + shell@hammerhead:/ $ ./hello + it worked!! + + ...okay, hi. hello world. whatever. + shell@hammerhead:/ $ exit + $ + +### How to add rust code into an ndk project to create an APK + +**Note: instructions beyond this point need further re-checking with latest android and rust.** + +Use rustc with --crate-type=staticlib etc to emit rust code which can be linked with C/C++ sources compiled by the android standalone toolchain's g++. From here is it possible to create an APK as with NDK examples. Use #[no_mangle] extern "C" rust functions to export functions which can be called by android frameworks + +Sample of modifications to the android-ndk "native-activity" sample makefile (jni/Android.mk) :- + + LOCAL_PATH := $(call my-dir) + include $(CLEAR_VARS) + + LOCAL_MODULE := rust-prebuilt + LOCAL_SRC_FILES := librust_android.a + + include $(PREBUILT_STATIC_LIBRARY) + include $(CLEAR_VARS) + + LOCAL_MODULE := native-activity + LOCAL_SRC_FILES := main.c + LOCAL_LDLIBS := -llog -landroid -lEGL -lGLESv1_CM + LOCAL_STATIC_LIBRARIES := android_native_app_glue rust-prebuilt + + include $(BUILD_SHARED_LIBRARY) + + $(call import-module,android/native_app_glue) + +This requires that you compiled rust code into a library beforehand, eg if 'hello_android.rs' is your crate root + + rustc --target=arm-linux-androideabi hello_android.rs --android-cross-path=/opt/ndk-standalone-arm/ --staticlib -o jni/librust_android.a + +execute ndk-build, ant debug|release etc to compile it into a native code .so and package,deploy it + +### How to integrate as a Shared Library into Android Studio using NDK and Gradle +1. Edit your Cargo.toml to build as a dylib + + [lib] + name = "my-awesome-lib" + crate_type = ["dylib"] +2. Create a `jni` and `jniLibs` directory in your Android Studio project in `proj/app/src/main` +3. Copy your `libmy-awesome-lib-SOME_HASH.so` to `proj/app/src/main/jniLibs` without the hash extension cargo automagically generates +4. Place your C++ shim source in `proj/app/src/main/jni` +5. Since we have no headers to register symbols, you will need to register all of the `#[no_mangle] pub extern` Rust functions in your C++ shim before you call them. + + e.g.) + + Rust + ----- + #[no_mangle] + pub extern an_int() -> c_int { + 123 as c_int + } + + C++ Shim + --------- + extern "C" { + + // Rust function prototypes + int an_int(); + + jint + Java_com_namespace_appname_MainActivity_callRustFunction(JNIEnv* env, jobject obj) + { + int some_int = an_int(); + return some_int; + } + + } // End extern + +6. Android Studio will automatically include all `.so` files in `jniLibs` in the APK, but if you want to use a custom shim to wrap the Rust code in order to get the JNI name mangling correct, you will need a custom `Android.mk` with your shim source in your `jni` directory: + + LOCAL_PATH := $(call my-dir) + + include $(CLEAR_VARS) + LOCAL_MODULE := my-awesome-lib + LOCAL_SRC_FILES := ../jniLibs/$(TARGET_ARCH_ABI)/libmy-awesome-lib.so + include $(PREBUILT_SHARED_LIBRARY) + + include $(CLEAR_VARS) + LOCAL_MODULE := app-name + LOCAL_SRC_FILES := shim.cpp + LOCAL_SHARED_LIBRARIES := my-awesome-lib + include $(BUILD_SHARED_LIBRARY) +7. By default, Android Studio generates a makefile in order to link against libs and headers included with the NDK and will generate one for all of your code in the `jni` directory. Unfortunately, this will not allow us to use ours, which includes our shared lib. A custom build.gradle will need created in order to manually build using `Android.mk` and copy the generated libs into the `jniLibs` directory. + +```sh +import org.apache.tools.ant.taskdefs.condition.Os + +def getPlatformNdkBuildCommand() { + def rootDir = project.rootDir + def localProperties = new File(rootDir, "local.properties") + if (localProperties.exists()) { + Properties properties = new Properties() + localProperties.withInputStream { + instr -> properties.load(instr) + } + def ndkDir = properties.getProperty('ndk.dir') + if (ndkDir == null) { + throw new GradleException("The ndk.dir property in local.propeties is not set") + } + def ndkBuild = Os.isFamily(Os.FAMILY_WINDOWS) ? "$ndkDir/ndk-build.cmd" : "$ndkDir/ndk-build" + return ndkBuild + } else { + throw new GradleException("The local.properties file does not exist") + } + } + + apply plugin: 'com.android.application' + + android { + compileSdkVersion 21 + buildToolsVersion "21.1.2" + + sourceSets.main { + jniLibs.srcDir 'src/main/jniLibs' + jni.srcDirs = [] //disable automatic ndk-build call + } + + defaultConfig { + applicationId "com.namespace.appname" + minSdkVersion 15 + targetSdkVersion 21 + versionCode 1 + versionName "1.0" + } + + buildTypes { + release { + minifyEnabled false + proguardFiles getDefaultProguardFile('proguard-android.txt'), 'proguard-rules.pro' + } + } + + // Compile cpp code from Android.mk + task buildCppShim(type: Exec) { + def ndkBuild = getPlatformNdkBuildCommand() + commandLine "$ndkBuild", '-j8', '-C', file('src/main/jni').absolutePath + } + + // Copy shared libs into jniLibs folder (Hacky workaround) + task copySharedLibs(type: Copy) { + from 'src/main/libs' + into 'src/main/jniLibs' + } + + tasks.withType(JavaCompile) { + compileTask -> compileTask.dependsOn buildCppShim + } + + copySharedLibs.dependsOn buildCppShim + copySharedLibs.execute() + } + + dependencies { + compile fileTree(dir: 'lib', include: ['*.jar']) + compile 'com.android.support:appcompat-v7:21.0.3' + } +``` + +8. Add the location of the NDK to your `local.properties` file + + ndk.dir=/abs/path/to/extracted/android-ndk-r10d +9. Load the libraries in your Activity + + static { + System.loadLibrary("my-awesome-lib"); + System.loadLibrary("cpp-shim"); + } + +## iOS + +You need XCode 5. + +Build Rust cross-compiler: +``` +mkdir build_ios; cd build_ios +../configure --target=armv7-apple-ios,armv7s-apple-ios,i386-apple-ios,aarch64-apple-ios,x86_64-apple-ios +make +``` + +Once compilation complete you can use it. + +To target device (depending on dest architecture): +``` +rustc --target=armv7-apple-ios foo.rs +``` + +``` +rustc --target=armv7s-apple-ios foo.rs +``` + +``` +rustc --target=aarch64-apple-ios foo.rs +``` + +To target simulator: +``` +rustc --target=i386-apple-ios foo.rs +``` + +``` +rustc --target=x86_64-apple-ios foo.rs +``` + +## What you get + +* all Rust superpowers +* exception handling +* LLDB debugging + +## Known limitations + +* segmented stacks are disabled which means no stack protection available + +## Resources + +* [Sample](https://github.com/vhbit/ObjCrust) of using Rust from Objective C + Makefile for simultaneous compiling for both device and simulator (all architectures)