在嵌入式 Linux 设备上使用 Flutter 开发图形界面

By Toradex胡珊逢

在嵌入式设备上开发图形用户界面通常会选择 Qt。这是一种经验证的方案,我们可以在多个领域看到用 Qt 开发的 UI。但随着移动端和 web 端界面更为广泛地使用,源自于这两个领域的技术也开始向嵌入式设备渗透。Flutter 就是一个例子。本文将介绍如何在Torizon 平台上如何使用Flutter 来开发用户界面。


Google 面向Android, iOS 推出的跨平台移动应⽤开发框架Flutter 可以构建高质量的原⽣⽤户界⾯,并可以扩展支持Web 和桌面应用。Flutter 尚未官方支持嵌入式系统,但目前 Sony 和 Ubuntu 正在致力于该工作。例如来自Sony 的elinux 可以在嵌入式平台上使用 Flutter。我们也将以此为基础,在 Verdin iMX8M Plus 的 Torizon 上运行Flutter 应用。


我们来看看Flutter 的构架。如下图所示,其由三个部分构成,User app, Framework 和Engine。flutter-elinux-linux 属于为嵌入式提供支持的embbedder。它可以运行在wayland 显示后台,这也是Torizon 提供的显示框架。flutter-elinux-linux 将提供flutter-client ,libflutter_elinux_wayland.so 和libflutter_engine.so。这些软件的功能参考该网页描述。


https://github.com/sony/flutter-embedded-linux/wiki/Features-of-Embedded-Linux-embedding-for-Flutter


flutter-elinux 是Flutter SDK 的一个非官方插件,用于为嵌入式设备创建、编译和调试Flutter 应用,并使用flutter-elinux-linux 在设备上显示。


为了减少对编译电脑的软件环境影响,我们将使用docker 容器进行编译。使用下面命令获取ubuntu:20.04 容器并进入其中。由于后面需要向容器内提供文件,这里将/home/user/flutter 映射到容器内的/opt/flutter。

-----------------------------------------------

~$ docker pull ubuntu:20.04

~$ docker run -it -v /home/user/flutter:/opt/flutter --name flutter_build ubuntu:20.04 /bin/bash

-----------------------------------------------


如果后续进入该容器重新编译,可以使用下面命令:

-----------------------------------------------

~$ docker container start flutter_build

~$ docker exec -it flutter_build bash

-----------------------------------------------


在容器中安装所需的软件。

-----------------------------------------------

# apt update

# apt upgrade

# apt install clang cmake build-essential pkg-config libegl1-mesa-dev libxkbcommon-dev libgles2-mesa-dev

# apt install libwayland-dev wayland-protocols git curl wget unzip git

# apt install python2

# apt install virtualenv

-----------------------------------------------


下载编译工具。

-----------------------------------------------

# mkdir -p /opt/flutter

# cd /opt/flutter

# git clone https://chromium.googlesource.com/chromium/tools/depot_tools.git

# export PATH=$PATH:$(pwd)/depot_tools

-----------------------------------------------


默认的 ubuntu:20.04 使用 Python3,在容器里使用 virtualenv 创建 Python2 环境。

-----------------------------------------------

# virtualenv .env -p python2

# source .env/bin/activate

-----------------------------------------------


创建 .gclient 文件并指定版本。

-----------------------------------------------

# cat .gclient

solutions = [

{

  "managed": False,

  "name": "src/flutter",

  "url": "https://github.com/flutter/engine.git@bd539267b42051b0da3d16ffa8f48949dce8aa8f",

  "custom_deps": {},

  "deps_file": "DEPS",

  "safesync_url": "",

  "custom_vars" : {

    "download_android_deps" : False,

    "download_windows_deps" : False,

  },

},

]

-----------------------------------------------


上面的 bd539267b42051b0da3d16ffa8f48949dce8aa8f 对应 ${path_to_flutter_sdk_install}/flutter/bin/internal/engine.version,两者需要一致。如果不指定的话,会下载最新的版本。除非确实需要编译最新版本的 Engine,否则并不推荐。

获取代码。

-----------------------------------------------

# gclient sync

-----------------------------------------------


编译 embbedder。这里编译为 arm64 目标 release 模式的 embedder。

-----------------------------------------------

# cd src

# ./flutter/tools/gn --target-os linux --linux-cpu arm64 --runtime-mode release --embedder-for-target --disable-desktop-embeddings --no-build-embedder-examples

# ninja -C out/linux_release_arm64

-----------------------------------------------


编译成功后在 out/linux_release_arm64 目录中可以看到 libflutter_engine.so 文件。


接下来将编译 Embedded Linux embedding for Flutter,这里会生成 flutter-client 和 libflutter_elinux_wayland.so。如果在X86 电脑上交叉编译需要使用Yocoto Project 生成包含clang 在内的SDK,这会涉及大量的工作内容。在Torizon 中我们可以直接使用debian 容器并通过apt 命令安装相应的软件,在Verdin iMX8M Plus 本地编译。这通常适用于代码量不是很多的项目。在Verdin iMX8M Plus 上运行下面命令启动debian 容器。

-----------------------------------------------

~$ docker run -it  -v /home/torizon/workspace:/opt/workspace torizon/debian:$CT_TAG_DEBIAN /bin/bash

-----------------------------------------------


在容器中安装所需的软件。

-----------------------------------------------

# apt update

# apt install clang cmake build-essential pkg-config libegl1-mesa-dev libxkbcommon-dev libgles2-mesa-dev

# apt install unzip git

# apt install curl wget

# apt install libwayland-dev wayland-protocols

# apt install libdrm-dev libgbm-dev libinput-dev libudev-dev libsystemd-dev

# cd /opt/workspace

-----------------------------------------------


下载 flutter-embedded-linux 代码。

-----------------------------------------------

# git clone https://github.com/sony/flutter-embedded-linux.git

# cd flutter-embedded-linux/

# mkdir build

-----------------------------------------------


此时将刚才编译的 libflutter_engine.so 复制到 build 目录。然后分别执行下面两个命令。

-----------------------------------------------

# cmake -DUSER_PROJECT_PATH=examples/flutter-wayland-client -DCMAKE_BUILD_TYPE=Release ..


# cmake -DUSER_PROJECT_PATH=examples/flutter-wayland-client -DCMAKE_BUILD_TYPE=Release -DBUILD_ELINUX_SO=ON -DBACKEND_TYPE=WAYLAND -DENABLE_ELINUX_EMBEDDER_LOG=OFF -DFLUTTER_RELEASE=ON ..

-----------------------------------------------


编译完成后在 build 目录下可以看到生成的 flutter-client 和 libflutter_elinux_wayland.so 两个文件。上面使用的编译选项含义请参考该网页说明。


接下来重新回到 X86 编译电脑开始 Flutter 应用的编译。用下面命令重新进入使之前的 ubuntu:20.04 容器。

-----------------------------------------------

~$ docker exec -it flutter_build bash

-----------------------------------------------



下载 flutter-elinux。这个是 Flutter SDK。

-----------------------------------------------

# cd /opt/flutter/

# git clone https://github.com/sony/flutter-elinux

# export PATH=$PATH:/opt/flutter/flutter-elinux/bin

-----------------------------------------------


运行下面命令查看安装情况。

-----------------------------------------------

# flutter-elinux doctor

# flutter-elinux devices

-----------------------------------------------


创建一个示例工程。

-----------------------------------------------

# flutter-elinux create demo1

# cd demo1

-----------------------------------------------


按照这里的说明交叉编译创建的工程。但在这之前需要准备Verdin iMX8M Plus 的arm64 格式文件系统。该文件系统可以是来自刚才在Verdin iMX8M Plus 上编译Embedded Linux embedding for Flutter 的容器。


在Verdin iMX8M Plus 运行下面命令,查看容器ID。

-----------------------------------------------

$ docker ps -a

CONTAINER ID        IMAGE                       COMMAND             CREATED             STATUS                     PORTS               NAMES

3dea07245b24        torizon/debian:2-bullseye   "/bin/bash"         2 days ago          Exited (137) 2 hours ago                       hardcore_nightingale

-----------------------------------------------


将容器的文件系统复制出来并打包。

-----------------------------------------------

$ sudo docker cp 3dea07245b24:/ arm64-sysroot

$ sudo tar cvf arm64-sysroot.tar arm64-sysroot

-----------------------------------------------


然后将 arm64-sysroot.tar 复制到 X86 编译电脑的 flutter_build 容器中,位于 /opt/flutter 目录。回到 flutter_build 容器,解压 arm64-sysroot.tar。

-----------------------------------------------

# cd /opt/flutter/

# tar vxf arm64-sysroot.tar

-----------------------------------------------


进入刚才创建的 demo1 目录,运行下面命令编译。

-----------------------------------------------

# cd demo1

# flutter-elinux build elinux --target-arch=arm64 --target-sysroot=/opt/flutter/arm64-sysroot

-----------------------------------------------


待编译结束后,查看 build/elinux/arm64/release/bundle,这里是 Flutter app 运行所需的所以文件。

-----------------------------------------------

# tree build/elinux/arm64/release/bundle -L 2

.

|-- data

|   |-- flutter_assets

|   `-- icudtl.dat

|-- demo1

`-- lib

  |-- libapp.so

  |-- libflutter_elinux_wayland.so

  `-- libflutter_engine.so

-----------------------------------------------


libflutter_elinux_wayland.so 和 libflutter_engine.so 是 Fltter SDK 预编译的库文件,需要将其替换为之前编译的库文件。


将 flutter-client 和 bundle 文件夹复制到 Verdin iMX8M Plus 的 /home/torizon/flutter_demo 目录。然后先启动weston 容器。

-----------------------------------------------

$ docker run -e ACCEPT_FSL_EULA=1 -d --rm --name=weston --net=host --cap-add CAP_SYS_TTY_CONFIG \

           -v /dev:/dev -v /tmp:/tmp -v /run/udev/:/run/udev/ \

           --device-cgroup-rule='c 4:* rmw' --device-cgroup-rule='c 13:* rmw' \

           --device-cgroup-rule='c 199:* rmw' --device-cgroup-rule='c 226:* rmw' \

           torizon/weston-vivante:$CT_TAG_WESTON_VIVANTE --developer weston-launch \

           --tty=/dev/tty7 --user=torizon

-----------------------------------------------


再启动另外一个 torizon/weston-vivante 容器,在里面我们将用命令行的方式启动编译好的 demo1。

-----------------------------------------------

$ docker run -e ACCEPT_FSL_EULA=1 -it --rm --name=wayland-app --user=torizon \

           -v /dev/dri:/dev/dri -v /dev/galcore:/dev/galcore -v /tmp:/tmp -v /home/torizon/flutter_demo:/opt/flutter \

           --device-cgroup-rule='c 199:* rmw' --device-cgroup-rule='c 226:* rmw' \

           torizon/weston-vivante:$CT_TAG_WESTON_VIVANTE bash

-----------------------------------------------


在启动的容器内运行下面命令。

-----------------------------------------------

# cd /opt/flutter

# LD_LIBRARY_PATH=/opt/flutter/bundle/lib/ ./flutter-client -b /opt/flutter/bundle

-----------------------------------------------






最后,我们将介绍如何导入一个现成的 Flutter 项目并打包为一个容器,用 docker-compose 文件启动。这里以 covid19_mobile_app为例进行说明。


在 flutter_build 容器中,下载 covid19_mobile_app 代码,将之前 demo1 目录中的 elinux 文件夹复制到 covid19_mobile_app 后再编译。同样,libflutter_elinux_wayland.so 和libflutter_engine.so 也需要替换为之前编译的库文件。

-----------------------------------------------

# cp -r ../demo1/elinux covid19_mobile_app

# flutter-elinux pub get

# flutter-elinux build elinux --target-arch=arm64 --target-sysroot=/opt/flutter/arm64-sysroot

-----------------------------------------------


编译结束后,将 covid19_mobile_app 的 bundle 目录连同 flutter-client,以及下面的 startup.sh,Dockerfile 放到任一目录中。


startup.sh

-----------------------------------------------

#!/bin/bash

LD_LIBRARY_PATH=/home/torizon/bundle/lib/ /usr/sbin/flutter-client -b /home/torizon/bundle

-----------------------------------------------


Dockerfile

-----------------------------------------------

FROM --platform=linux/arm64 torizon/weston-vivante:2

ADD bundle /home/torizon/bundle

COPY flutter-client /usr/sbin

COPY startup.sh /home/torizon

CMD [ "/home/torizon/startup.sh" ]

-----------------------------------------------


运行下面命令生成一个 flutter app 容器。

-----------------------------------------------

$ docker build -t flutter_demo:1 .

-----------------------------------------------


将 flutter_demo:1 容器和 docker-compose.yml 文件复制到 Verdin iMX8M Plus 上。


docker-compose.yml

-----------------------------------------------

services:

flutter_demo_covid19:

  depends_on:

  - weston

  devices: []

  image: flutter_demo:1

  ports: []

  device_cgroup_rules:

  - c 199:* rmw

  - c 226:* rmw

  volumes:

  - /tmp:/tmp:rw

  - /dev/dri:/dev/dri:rw

  - /dev/galcore:/dev/galcore:rw

weston:

  cap_add:

  - CAP_SYS_TTY_CONFIG

  device_cgroup_rules:

  - c 4:0 rmw

  - c 4:7 rmw

  - c 13:* rmw

  - c 199:* rmw

  - c 226:* rmw

  environment:

  - ACCEPT_FSL_EULA=1

  image: torizon/weston-vivante:2

  network_mode: host

  volumes:

  - source: /tmp

    target: /tmp

    type: bind

  - source: /dev

    target: /dev

    type: bind

  - source: /run/udev

    target: /run/udev

    type: bind

version: '2.4'

-----------------------------------------------


运行 docker-compose up -d 即可启动 weston 和 flutter app 容器。





总结

Flutter 框架为图形界面开发提供一个非常灵活的方案,使得嵌入式开发也可以从中受益。于此同时,嵌入式 Linux 对 Flutter 的支持也处于早期阶段,项目开发需要充分验证。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,658评论 6 496
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,482评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,213评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,395评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,487评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,523评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,525评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,300评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,753评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,048评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,223评论 1 343
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,905评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,541评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,168评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,417评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,094评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,088评论 2 352

推荐阅读更多精彩内容