想了解HIDL介绍的可以参考《HIDL概述》,本篇文章主要介绍HIDL的详细编译流程及简单的客户端应用(C++跟Android客户端的应用)。

一、准备工作

整一套源码,Android O或者Android P的
保证全套代码已经全编译,原生代码全编译命令
source build/envset.sh
lunch
make
   3、hidl-gen工具已经安装,安装命令

make hidl-gen 
二、hidl-gen工具介绍

系统定义的所有的.hal接口,都是通过hidl-gen工具转换成对应的代码。hidl-gen源码路径:system/tools/hidl,是在ubuntu上可执行的二进制文件。

使用方法:hidl-gen -o output-path -L language (-r interface-root) fqname
例子:
hidl-gen -o hardware/interfaces/gunder/1.0/default/ 
     -Lc++-impl -randroid.hardware:hardware/interfaces 
     -randroid.hidl:system/libhidl/transport android.hardware.gunder@1.0


可以使用hidl-gen 查看帮助,如图1:

  

三、项目实例

1、在hardware/interfaces/目录下新建gunder/1.0目录,并在1.0目录中创建接口IGunder.hal。目录结构如下:

gunder@ubuntu-MX8:~ /IMX8_b0beta2/hardware/interfaces/gunder$ tree

└── 1.0

    └── IGunder.hal

IGunder.hal文件里面只有一个接口IGunder和一个方法helloWorld(string name),具体实现如下:

package android.hardware.gunder@1.0;
 
interface IGunder{
    helloWorld(string name) generates (string result);
};
 

  2、执行下面三条命令会自动生成对应的c++文件;

PACKAGE=android.hardware.gunder@1.0
LOC=hardware/interfaces/gunder/1.0/default/
      hidl-gen -o $LOC -Lc++-impl -randroid.hardware:hardware/interfaces 
     -randroid.hidl:system/libhidl/transport $PACKAGE
参数说明:

-L: 语言类型,包括c++, c++-headers, c++-sources, export-header, c++-impl, 
     java, java-constants, vts, makefile, androidbp, androidbp-impl, hash等。
     hidl-gen可根据传入的语言类型产生不同的文件。
fqname: 完全限定名称的输入文件。比如本例中android.hardware.gunder@1.0,
     要求在源码目录下必须有hardware/interfaces/ gunder /1.0/目录。
     对于单个文件来说,格式如下:package@version::fileName,
     比如android.hardware. gunder @1.0::types.Feature。
     对于目录来说。格式如下package@version,
     比如android.hardware. gunder @1.0。
     -r: 格式package:path,可选,
     对fqname对应的文件来说,用来指定包名和文件所在的目录到Android系统源码根目录的路径。
     如果没有制定,前缀默认是:android.hardware,目录是Android源码的根目录。
-o:存放hidl-gen产生的中间文件的路径。

 

执行命令后的目录结构如下:

gunder@ubuntu-MX8:~ /IMX8_b0beta2$ PACKAGE=android.hardware.gunder@1.0
gunder@ubuntu-MX8:~ /IMX8_b0beta2$ LOC=hardware/interfaces/gunder/1.0/default/
gunder@ubuntu-MX8:~ /IMX8_b0beta2$ hidl-gen -o $LOC -Lc++-impl -randroid.hardware:hardware/interfaces -randroid.hidl:system/libhidl/transport $PACKAGE
gunder@ubuntu-MX8:~ /IMX8_b0beta2/hardware/interfaces/gunder$ tree

└── 1.0

    ├── default

    │   ├── Gunder.cpp

    │   └── Gunder.h

    └── IGunder.hal

default 是新生成的目录,Gunder.cpp和Gunder.h是新生成的两个文件,打开Gunder.h文件,去掉// extern "C" IGunder* HIDL_FETCH_IGunder(const char* name);前面的注释,使用直通式HAL(Passthrough 模式)来通信。Gunder.h文件修改后如图2:

         

Gunder.cpp文件也要进行对应的修改,修改后如如图3:  

3、执行下面命令

hidl-gen -o $LOC -Landroidbp-impl 
     -randroid.hardware:hardware/interfaces 
     -randroid.hidl:system/libhidl/transport $PACKAGE

会在hardware/interfaces/gunder/1.0/default目录生成Android.bp文件。

gunder@ubuntu-MX8:~ /IMX8_b0beta2/hardware/interfaces/gunder$ tree

└── 1.0

    ├── default

    │   ├── Android.bp

    │   ├── Gunder.cpp

    │   └── Gunder.h

└── IGunder.hal

4、然后使用脚本update-makefiles.sh来更新Makefile,自动在hardware/interfaces/gunder/1.0目录生成Android,mk和 Android.bp,hardware/interfaces/gunder目录生成Android.bp。命令如下

./hardware/interfaces/update-makefiles.sh

执行完命令后目录结构如下:

gunder@ubuntu-MX8:~/IMX8_b0beta2$ ./hardware/interfaces/update-makefiles.sh
gunder@ubuntu-MX8:~/IMX8_b0beta2/hardware/interfaces/gunder$ tree.

├── 1.0

│   ├── Android.bp

│   ├── Android.mk

│   ├── default

│   │   ├── Android.bp

│   │   ├── Gunder.cpp

│   │   └── Gunder.h

│   └── IGunder.hal

└── Android.bp

5、在hardware/interfaces/gunder/1.0/default目录下新建service.cpp跟android.hardware.gunder@1.0-service.rc,其中android.hardware.gunder@1.0-service.rc是程序的入口函数。新的目录结构如下

├── 1.0

│   ├── Android.bp

│   ├── Android.mk

│   ├── default

│   │   ├── Android.bp

│   │   ├── android.hardware.gunder@1.0-service.rc

│   │   ├── Gunder.cpp

│   │   ├── Gunder.h

│   │   └── service.cpp

│   └── IGunder.hal

└── Android.bp

android.hardware.gunder@1.0-service.rc的实现如图:

service.cpp的实现如图:

打开hardware/interfaces/gunder/1.0/default目录下的Android.bp,添加编译service.cpp成为可执行文件的代码。具体添加内容如下:

到此跟HAL相关的代码就实现完了。

6、编译生成服务端跟客户端要用各种库文件。首先执行下面两条命令

./hardware/interfaces/update-makefiles.sh
mmm hardware/interfaces/gunder/1.0/

执行后会生成下面的文件:

out/target/product/mek_8q/system/lib64/android.hardware.gunder@1.0.so
out/target/product/mek_8q/vendor/lib64/hw/android.hardware.gunder@1.0-impl.so
out/target/product/mek_8q/vendor/etc/init/android.hardware.gunder@1.0-service.rc

7、在manifest文件里添加vendor接口的定义编译device/fsl/mek_8q/manifest.xml文件(不同厂家路径可能不一样),添加以下内容,不然在client端是没法拿到service的。如下:

8、使用C++实现客户端调用

在hardware/interfaces/gunder/1.0目录下新建test目录,并且在test目录下新建Android.bp跟GunderTest.cpp文件,这两个文件的内容如下:GunderTest.cpp:

Android.bp:

执行以下命令:

/hardware/interfaces/update-makefiles.sh
mmm hardware/interfaces/gunder/1.0/

执行第一条命令是为了更新hardware/interfaces/gunder/目录下的Android.bp文件,如下:

执行第二条命令会生成可执行文件:

out/target/product/mek_8q/vendor/bin/hw/gunder_client

9、使用JAVA实现客户端调用

 为了方便eclipse或者Android Studio调用接口函数,需要编译出classes.jar包。但是jack编译出来的文件是classes.jack。如图:

为了编译出classes.jar,需要打开hardware/interfaces/gunder/1.0目录的Android.mk,在include $(CLEAR_VARS)下面添加LOCAL_JACK_ENABLED := disabled。这样编译的时候就不走jack编译了。如下:

执行下面命令

mmm hardware/interfaces/gunder/1.0/

执行完后生成了classes.jar。如图:

新建Android项目HIDLdemo,将classes.jar导入项目,MainActivity代码实现如下:

添加Android.mk文件,然后将项目放到packages/apps/进行编译。Android.mk文件内容如下:

执行命令

mmm packages/apps/HIDLdemo


会生成out/target/product/mek_8q/system/app/HIDLdemo/HIDLdemo.apk

 10、测试客户端程序

  执行下列命令:

adb push 
     X:\IMX8_b0beta2\out\target\product\mek_8q\vendor\lib64
     \hw\android.hardware.gunder@1.0-impl.so  /vendor/lib64/hw
adb push 
     X:\IMX8_b0beta2\out\target\product\mek_8q\system\lib64
     \android.hardware.gunder@1.0.so /system/lib64
adb push X:\IMX8_b0beta2\out\target\product\mek_8q\vendor\bin\hw
     \gunder_client /vendor/bin/hw
adb push 
    X:\IMX8_b0beta2\out\target\product\mek_8q\vendor\bin\hw
    \android.hardware.gunder@1.0-service /vendor/bin/hw
adb push X:\IMX8_b0beta2\device\fsl\evk_8mq\manifest.xml /vendor
adb install -r 
    X:\IMX8_b0beta2\out\target\product\ mek_8q \system\app\HIDLdemo\HIDLdemo.apk

上面的命令是将需要的文件push到系统,方便调试。

C++客户端调试:

先启动./android.hardware.gunder@1.0-service服务,然后启动gunder_client程序,执行结果如下:

Android客户端执行:

先启动./android.hardware.gunder@1.0-service服务,然后通过命令拉起MainActivity界面

adb shell am start -n com.example.ytkj.hidldemo/.MainActivity,执行结果如下:

Logo

开放原子开发者工作坊旨在鼓励更多人参与开源活动,与志同道合的开发者们相互交流开发经验、分享开发心得、获取前沿技术趋势。工作坊有多种形式的开发者活动,如meetup、训练营等,主打技术交流,干货满满,真诚地邀请各位开发者共同参与!

更多推荐