From 4f0c486bf7744188910b81182ec3a992af86bb34 Mon Sep 17 00:00:00 2001 From: Lucas Fryzek Date: Thu, 27 Jun 2024 16:34:07 +0100 Subject: Add Android swrast blog post --- html/assets/2024-06-27-android-swrast/lavapipe.png | Bin 0 -> 138410 bytes .../lavapipe_thumbnail.png | Bin 0 -> 17820 bytes notes/android_swrast.md | 88 +++++++++++++++++++++ 3 files changed, 88 insertions(+) create mode 100644 html/assets/2024-06-27-android-swrast/lavapipe.png create mode 100644 html/assets/2024-06-27-android-swrast/lavapipe_thumbnail.png create mode 100644 notes/android_swrast.md diff --git a/html/assets/2024-06-27-android-swrast/lavapipe.png b/html/assets/2024-06-27-android-swrast/lavapipe.png new file mode 100644 index 0000000..93918d2 Binary files /dev/null and b/html/assets/2024-06-27-android-swrast/lavapipe.png differ diff --git a/html/assets/2024-06-27-android-swrast/lavapipe_thumbnail.png b/html/assets/2024-06-27-android-swrast/lavapipe_thumbnail.png new file mode 100644 index 0000000..23048b5 Binary files /dev/null and b/html/assets/2024-06-27-android-swrast/lavapipe_thumbnail.png differ diff --git a/notes/android_swrast.md b/notes/android_swrast.md new file mode 100644 index 0000000..f4a3d76 --- /dev/null +++ b/notes/android_swrast.md @@ -0,0 +1,88 @@ +--- +title: Software Rendering and Android +date: "2024-06-27" +last_edit: "2024-06-27" +status: 3 +categories: igalia graphics +cover_image: "/assets/2024-06-27-android-swrast/lavapipe_thumbnail.png" +--- + +My current project at Igalia has had me working on Mesa's software renderers, llvmpipe and lavapipe. I've +been working to get them running on Android, and I wanted to document the progress I've made, the challenges +I've faced, and talk a little bit about the development process for a project like this. My work is not +totally merged into upstream mesa yet, but you can see the MRs I made here: + +- [MR !29344](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/29344) +- [MR !29785](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/29785) +- [MR !27805](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/27805) +- [MR !28735](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/28735) + +## Setting up an Android development environment + +Getting system level software to build and run on Android is unfortunately not straightforward. Since we are doing +software rendering we don't need a physical device and instead we can make use of the Android emulator, and if you +didn't know Android has two emulators, the common one most people use is "goldfish" and the other lesser known +is "cuttlefish". For this project I did my work on the cuttlefish emulator as its meant for testing the +Android OS itself instead of just Android apps and is more reflective of real hardware. The cuttlefish emulator +takes a little bit more work to setup, and I've found that it only works properly in Debian based linux distros. +I run Fedora, so I had to run the emulator in a debian VM. + +Thankfully Google has good instructions for building and running cuttlefish, which you can find [here](https://source.android.com/docs/devices/cuttlefish/get-started). +The instructions show you how to setup the emulator using nightly build images from Google. We'll also need to +setup our own Android OS images so after we've confirmed we can run the emulator, we need to start looking at +building AOSP. + +For building our own AOSP image, we can also follow the instructions from Google [here](https://source.android.com/docs/setup/build/building). +For the target we'll want `aosp_cf_x86_64_phone-trunk_staging-eng`. At this point its a good idea to verify that +you can build the image, you can follow the rest of the instructions on the page for building the image. Building +AOSP from source does take awhile though, so prepare to wait potentially an entire day for the image to build. +Also if you get errors complaining that you're out of memory, you can try to reduce the number of parallel builds. +Google officially recommends to have 64GB of RAM, and I only had 32GB so some packages had to be built with the +parallel builds set to 1 so I wouldn't run out of RAM. + +For running this custom build image on Cuttlefish, you can just copy all the `*.img` files from `out/target/product/vsoc_x86_64/` +to the root cuttlefish directory, and then launch cuttlefish. If everything worked successfully you should be +able to see your custom built AOSP image running in the cuttlefish webui. + +## Building Mesa targeting Android + +Working from the changes in MR [!29344](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/29344) +building llvmpipe or lavapipe targeting Android should just work™️. To get to that stage required a few +changes. First llvmpipe actually already had some support on Android, as long as it was running on a device +that supports a DRM display driver. In that case it could use the `dri` window system integration which already +work on Android. I wanted to get llvmpipe (and lavapipe) running without dri, so I had to add support for +Android in the `drisw` window system integration. + +To support Android in `drisw`, this mainly meant adding support for importing dmabuf as framebuffers. The +Android windowing system will provide us with a "gralloc" buffer which inside has a dmabuf fd that represents +the framebuffer. Adding support for importing dmabufs in drisw means we can import and begin drawing to these +frame buffers. Most the changes to support that can be found in [`drisw_allocate_textures`](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/src/gallium/frontends/dri/drisw.c#L405) +and the underlying changes to llvmpipe to support importing dmabufs in MR [!27805](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/27805). +The EGL android platform code also needed some changes to use the `drisw` window system code. Previously this +code would only work with true dri drivers, but with some small tweaks it was possible to get to have it +initialize the drisw window system and then using it for rendering if no hardware devices are available. + +For lavapipe the changes we a lot simpler. The android Vulkan loader requires your driver to have +`HAL_MODULE_INFO_SYM` symbol in the binary, so that got created and populated correctly, following other vulkan +drivers in Mesa like turnip. Then the image creation code had to be modified to support the +`VK_ANDROID_native_buffer` extension which allows the Android vulkan loader to create images using Android native +buffer handles. Under the hood this means getting the dmabuf fd from the native buffer handle. Thankfully mesa +already has some common code to handle this, so I could just use that. Some other small changes were also +necessary to address crashes and other failures that came up during testing. + +With the changes out of of the way we can now start building Mesa on Android. For this project I had to update +the Android documentation for Mesa to include steps for building LLVM for Android since the version Google ships +with the NDK is missing libraries that llvmpipe/lavapipe need to function. You can see the updated documentation +[here](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/drivers/llvmpipe.rst) +and [here](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/android.rst). +After sorting out LLVM, building llvmpipe/lavapipe is the same as building any other Mesa diver for Android, we +setup a cross file to tell meson how to cross compile and then we run meson. At this point you could manual modify +the Android image and copy these files to the vm, but I also wanted to support building a new AOSP image directly +including the driver. In order to do that you also have to rename the driver binaries to match android's naming +convention, and make sure SO_NAME matches as well. If you check out [this](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/android.rst?plain=1#L183) +section of the documentation I wrote, it covers how to do that. + +If you followed all of that you should have built an version of llvmpipe and lavapipe that you can run on +Android's cuttlefish emulator. + +![Android running lavapipe](/assets/2024-06-27-android-swrast/lavapipe.png) -- cgit v1.2.3 From bd0efd175ea326629a2c9dd343326171ed0eef4e Mon Sep 17 00:00:00 2001 From: Ricardo Garcia Date: Fri, 28 Jun 2024 09:45:01 +0000 Subject: Apply 7 suggestion(s) to 1 file(s) --- notes/android_swrast.md | 14 +++++++------- 1 file changed, 7 insertions(+), 7 deletions(-) diff --git a/notes/android_swrast.md b/notes/android_swrast.md index f4a3d76..27a8ecc 100644 --- a/notes/android_swrast.md +++ b/notes/android_swrast.md @@ -33,14 +33,14 @@ setup our own Android OS images so after we've confirmed we can run the emulator building AOSP. For building our own AOSP image, we can also follow the instructions from Google [here](https://source.android.com/docs/setup/build/building). -For the target we'll want `aosp_cf_x86_64_phone-trunk_staging-eng`. At this point its a good idea to verify that -you can build the image, you can follow the rest of the instructions on the page for building the image. Building -AOSP from source does take awhile though, so prepare to wait potentially an entire day for the image to build. +For the target we'll want `aosp_cf_x86_64_phone-trunk_staging-eng`. At this point it's a good idea to verify that +you can build the image, which you can do by following the rest of the instructions on the page. Building +AOSP from source does take a while though, so prepare to wait potentially an entire day for the image to build. Also if you get errors complaining that you're out of memory, you can try to reduce the number of parallel builds. Google officially recommends to have 64GB of RAM, and I only had 32GB so some packages had to be built with the parallel builds set to 1 so I wouldn't run out of RAM. -For running this custom build image on Cuttlefish, you can just copy all the `*.img` files from `out/target/product/vsoc_x86_64/` +For running this custom-built image on Cuttlefish, you can just copy all the `*.img` files from `out/target/product/vsoc_x86_64/` to the root cuttlefish directory, and then launch cuttlefish. If everything worked successfully you should be able to see your custom built AOSP image running in the cuttlefish webui. @@ -50,7 +50,7 @@ Working from the changes in MR [!29344](https://gitlab.freedesktop.org/mesa/mesa building llvmpipe or lavapipe targeting Android should just work™️. To get to that stage required a few changes. First llvmpipe actually already had some support on Android, as long as it was running on a device that supports a DRM display driver. In that case it could use the `dri` window system integration which already -work on Android. I wanted to get llvmpipe (and lavapipe) running without dri, so I had to add support for +works on Android. I wanted to get llvmpipe (and lavapipe) running without dri, so I had to add support for Android in the `drisw` window system integration. To support Android in `drisw`, this mainly meant adding support for importing dmabuf as framebuffers. The @@ -62,7 +62,7 @@ The EGL android platform code also needed some changes to use the `drisw` window code would only work with true dri drivers, but with some small tweaks it was possible to get to have it initialize the drisw window system and then using it for rendering if no hardware devices are available. -For lavapipe the changes we a lot simpler. The android Vulkan loader requires your driver to have +For lavapipe the changes were a lot simpler. The android Vulkan loader requires your driver to have `HAL_MODULE_INFO_SYM` symbol in the binary, so that got created and populated correctly, following other vulkan drivers in Mesa like turnip. Then the image creation code had to be modified to support the `VK_ANDROID_native_buffer` extension which allows the Android vulkan loader to create images using Android native @@ -75,7 +75,7 @@ the Android documentation for Mesa to include steps for building LLVM for Androi with the NDK is missing libraries that llvmpipe/lavapipe need to function. You can see the updated documentation [here](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/drivers/llvmpipe.rst) and [here](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/android.rst). -After sorting out LLVM, building llvmpipe/lavapipe is the same as building any other Mesa diver for Android, we +After sorting out LLVM, building llvmpipe/lavapipe is the same as building any other Mesa driver for Android: we setup a cross file to tell meson how to cross compile and then we run meson. At this point you could manual modify the Android image and copy these files to the vm, but I also wanted to support building a new AOSP image directly including the driver. In order to do that you also have to rename the driver binaries to match android's naming -- cgit v1.2.3 From dda4821b4a34309a812e01b05a7aecbe77ec199f Mon Sep 17 00:00:00 2001 From: Lucas Fryzek Date: Fri, 28 Jun 2024 10:47:40 +0100 Subject: android_swrast: Update capitalization to be consistent --- notes/android_swrast.md | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/notes/android_swrast.md b/notes/android_swrast.md index 27a8ecc..219c087 100644 --- a/notes/android_swrast.md +++ b/notes/android_swrast.md @@ -58,14 +58,14 @@ Android windowing system will provide us with a "gralloc" buffer which inside ha the framebuffer. Adding support for importing dmabufs in drisw means we can import and begin drawing to these frame buffers. Most the changes to support that can be found in [`drisw_allocate_textures`](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/src/gallium/frontends/dri/drisw.c#L405) and the underlying changes to llvmpipe to support importing dmabufs in MR [!27805](https://gitlab.freedesktop.org/mesa/mesa/-/merge_requests/27805). -The EGL android platform code also needed some changes to use the `drisw` window system code. Previously this +The EGL Android platform code also needed some changes to use the `drisw` window system code. Previously this code would only work with true dri drivers, but with some small tweaks it was possible to get to have it initialize the drisw window system and then using it for rendering if no hardware devices are available. -For lavapipe the changes were a lot simpler. The android Vulkan loader requires your driver to have -`HAL_MODULE_INFO_SYM` symbol in the binary, so that got created and populated correctly, following other vulkan +For lavapipe the changes were a lot simpler. The Android Vulkan loader requires your driver to have +`HAL_MODULE_INFO_SYM` symbol in the binary, so that got created and populated correctly, following other Vulkan drivers in Mesa like turnip. Then the image creation code had to be modified to support the -`VK_ANDROID_native_buffer` extension which allows the Android vulkan loader to create images using Android native +`VK_ANDROID_native_buffer` extension which allows the Android Vulkan loader to create images using Android native buffer handles. Under the hood this means getting the dmabuf fd from the native buffer handle. Thankfully mesa already has some common code to handle this, so I could just use that. Some other small changes were also necessary to address crashes and other failures that came up during testing. @@ -78,7 +78,7 @@ and [here](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab After sorting out LLVM, building llvmpipe/lavapipe is the same as building any other Mesa driver for Android: we setup a cross file to tell meson how to cross compile and then we run meson. At this point you could manual modify the Android image and copy these files to the vm, but I also wanted to support building a new AOSP image directly -including the driver. In order to do that you also have to rename the driver binaries to match android's naming +including the driver. In order to do that you also have to rename the driver binaries to match Android's naming convention, and make sure SO_NAME matches as well. If you check out [this](https://gitlab.freedesktop.org/mesa/mesa/-/blob/9705df53408777d493eab19e5a58c432c1e75acb/docs/android.rst?plain=1#L183) section of the documentation I wrote, it covers how to do that. -- cgit v1.2.3 From 9d239a0296a572c26b2967319e17f8aca4662e58 Mon Sep 17 00:00:00 2001 From: Lucas Fryzek Date: Mon, 1 Jul 2024 08:22:30 +0100 Subject: android_swrast: Add references section with links --- notes/android_swrast.md | 13 +++++++++++++ 1 file changed, 13 insertions(+) diff --git a/notes/android_swrast.md b/notes/android_swrast.md index 219c087..67087ed 100644 --- a/notes/android_swrast.md +++ b/notes/android_swrast.md @@ -86,3 +86,16 @@ If you followed all of that you should have built an version of llvmpipe and lav Android's cuttlefish emulator. ![Android running lavapipe](/assets/2024-06-27-android-swrast/lavapipe.png) + +## References + +- + - Main MR with Android changes +- + - Google's official guide for getting started with the Cuttlefish emulator +- + - Google's official guide for building AOSP images +- + - My updated documentation in MR for llvmpipe +- + - My updated documentation in MR for Android integration in mesa -- cgit v1.2.3