android_packages_modules_Vi.../microdroid
Jooyung Han 6351310cfa apex: use the same key for all microdroid items
The pubkey embedded in bootloader should match with the key signing
VBmeta. The updated build graph is to ensure bootloader and VBmeta to be
generated with the same key.

All other filesystem images are signed with the same key for
convenience even though it's not necessary.

Bug: 193504286
Bug: 203726593
Test: atest MicrodroidHostTestCases
Change-Id: I4ecb9e2c00d739aba84677036edf256f141767ac
2021-11-08 10:45:42 +09:00
..
keymint Update for secure deletion support. 2021-08-11 06:40:15 -06:00
payload virtualizationservice: "prefer_staged" for apexes 2021-10-05 20:03:12 +09:00
Android.bp apex: use the same key for all microdroid items 2021-11-08 10:45:42 +09:00
README.md Clean up READMEs and remove unused target. 2021-08-16 12:19:16 +00:00
bootconfig.app_debuggable Run logd only when debug_level is set 2021-10-27 01:34:39 +09:00
bootconfig.arm64 Use bootconfig 2021-07-05 15:42:40 +09:00
bootconfig.common Use bootconfig 2021-07-05 15:42:40 +09:00
bootconfig.full_debuggable Run logd only when debug_level is set 2021-10-27 01:34:39 +09:00
bootconfig.normal Run logd only when debug_level is set 2021-10-27 01:34:39 +09:00
bootconfig.x86_64 Add one more virtio-console device 2021-11-01 20:07:15 +09:00
build.prop Don't start adb if debugging is off 2021-10-18 20:38:23 +09:00
dummy_dtb.img
empty_kernel
fstab.microdroid microdroid: use a single vbmeta for all partitions 2021-10-26 12:11:26 +09:00
init.rc Add one more virtio-console device 2021-11-01 20:07:15 +09:00
linker.config.json
microdroid.json microdroid: use a single vbmeta for all partitions 2021-10-26 12:11:26 +09:00
microdroid_compatibility_matrix.xml
microdroid_manifest.xml
microdroid_vendor_compatibility_matrix.xml
microdroid_vendor_manifest.xml
uboot-env-x86_64.txt Use the new verified_boot_android command 2021-09-04 01:46:41 +09:00
uboot-env.txt Use the new verified_boot_android command 2021-09-04 01:46:41 +09:00
ueventd.rc Add one more virtio-console device 2021-11-01 20:07:15 +09:00

README.md

Microdroid

Microdroid is a (very) lightweight version of Android that is intended to run on on-device virtual machines. It is built from the same source code as the regular Android, but it is much smaller; no system server, no HALs, no GUI, etc. It is intended to host headless & native workloads only.

Prerequisites

Any 64-bit target (either x86_64 or arm64) is supported. 32-bit target is not supported. Note that we currently don't support user builds; only userdebug builds are supported.

The only remaining requirement is that com.android.virt APEX has to be pre-installed. To do this, add the following line in your product makefile.

$(call inherit-product, packages/modules/Virtualization/apex/product_packages.mk)

Build the target after adding the line, and flash it. This step needs to be done only once for the target.

If you are using yukawa (VIM3L) or aosp_cf_x86_64_phone (Cuttlefish), adding above line is not necessary as it's already done.

Instructions for building and flashing Android for yukawa can be found here.

Building and installing microdroid

Microdroid is part of the com.android.virt APEX. To build it and install to the device:

banchan com.android.virt aosp_arm64
m apps_only dist
adb install out/dist/com.android.virt.apex
adb reboot

If your target is x86_64 (e.g. aosp_cf_x86_64_phone), replace aosp_arm64 with aosp_x86_64.

Building an app

An app in microdroid is a shared library file embedded in an APK. The shared library should have an entry point android_native_main as shown below:

extern "C" int android_native_main(int argc, char* argv[]) {
  printf("Hello Microdroid!\n");
}

Then build it as a shared library:

cc_library_shared {
  name: "MyMicrodroidApp",
  srcs: ["**/*.cpp"],
  sdk_version: "current",
}

Then you need a configuration file in JSON format that defines what to load and execute in microdroid. The name of the file can be anything and you may have multiple configuration files if needed.

{
  "os": { "name": "microdroid" },
  "task": {
    "type": "microdroid_launcher",
    "command": "MyMicrodroidApp.so"
  }
}

The value of task.command should match with the name of the shared library defined above. If your app requires APEXes to be imported, you can declare the list in apexes key like following.

{
  "os": ...,
  "task": ...,
  "apexes": [
    {"name": "com.android.awesome_apex"}
  ]
}

Embed the shared library and the VM configuration file in an APK:

android_app {
  name: "MyApp",
  srcs: ["**/*.java"], // if there is any java code
  jni_libs: ["MyMicrodroidApp"],
  use_embedded_native_libs: true,
  sdk_version: "current",
}

// The VM configuration file can be embedded by simply placing it at `./assets`
// directory.

Finally, you build and sign the APK.

TARGET_BUILD_APPS=MyApp m apps_only dist
m apksigner
apksigner sign --ks path_to_keystore out/dist/MyApp.apk

path_to_keystore should be replaced with the actual path to the keystore, which can be created as follows:

keytool -keystore my_keystore -genkey -alias my_key

Make sure that .apk.idsig file is also generated in the same directory as the signed APK.

Running the app on microdroid

First of all, install the signed APK to the target device.

adb install out/dist/MyApp.apk

ALL_CAPs below are placeholders. They need to be replaced with correct values:

  • VM_CONFIG_FILE: the name of the VM config file that you embedded in the APK. (e.g. vm_config.json)
  • PACKAGE_NAME_OF_YOUR_APP: package name of your app (e.g. com.acme.app).
  • PATH_TO_YOUR_APP: path to the installed APK on the device. Can be obtained via the following command.
    adb shell pm path PACKAGE_NAME_OF_YOUR_APP
    
    It shall report a cryptic path similar to /data/app/~~OgZq==/com.acme.app-HudMahQ==/base.apk.

Push idsig of the APK to the device.

TEST_ROOT=/data/local/tmp/virt
adb shell mkdir $TEST_ROOT
adb push out/dist/MyApp.apk.idsig $TEST_ROOT/

Execute the following commands to launch a VM. The VM will boot to microdroid and then automatically execute your app (the shared library MyMicrodroidApp.so).

TEST_ROOT=/data/local/tmp/virt
adb shell /apex/com.android.virt/bin/vm run-app \
--log $TEST_ROOT/log.txt \
PATH_TO_YOUR_APP \
$TEST_ROOT/MyApp.apk.idsig \
$TEST_ROOT/instance.img \
assets/VM_CONFIG_FILE

The last command lets you know the CID assigned to the VM. The console output from the VM is stored to $TEST_ROOT/log.txt file for debugging purpose. If you omit the --log $TEST_ROOT/log.txt option, it will be emitted to the current console.

Stopping the VM can be done as follows:

adb shell /apex/com.android.virt/bin/vm stop $CID

, where $CID is the reported CID value. This works only when the vm was invoked with the --daemonize flag. If the flag was not used, press Ctrl+C on the console where the vm run-app command was invoked.

ADB

On userdebug builds, you can have an adb connection to microdroid. To do so,

adb forward tcp:8000 vsock:$CID:5555
adb connect localhost:8000

$CID should be the CID that vm reported upon execution of the vm run command in the above. You can also check it with adb shell "/apex/com.android.virt/bin/vm list". 5555 must be the value. 8000 however can be any port on the development machine.

Done. Now you can log into microdroid. Have fun!

$ adb -s localhost:8000 shell