You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

README.md 14 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317
  1. # MindSpore Lite 端侧风格迁移demo(Android)
  2. 本示例程序演示了如何在端侧利用MindSpore Lite API以及MindSpore Lite风格迁移模型完成端侧推理,根据demo内置的标准图片更换目标图片的艺术风格,并在App图像预览界面中显示出来。
  3. ## 运行依赖
  4. - Android Studio >= 3.2 (推荐4.0以上版本)
  5. ## 构建与运行
  6. 1. 在Android Studio中加载本示例源码。
  7. ![start_home](images/home.png)
  8. 启动Android Studio后,点击`File->Settings->System Settings->Android SDK`,勾选相应的`SDK Tools`。如下图所示,勾选后,点击`OK`,Android Studio即可自动安装SDK。
  9. ![start_sdk](images/sdk_management.jpg)
  10. > Android SDK Tools为默认安装项,取消`Hide Obsolete Packages`选框之后可看到。
  11. >
  12. > 使用过程中若出现问题,可参考第4项解决。
  13. 2. 连接Android设备,运行该应用程序。
  14. 通过USB连接Android手机。待成功识别到设备后,点击`Run 'app'`即可在您的手机上运行本示例项目。
  15. > 编译过程中Android Studio会自动下载MindSpore Lite、模型文件等相关依赖项,编译过程需做耐心等待。
  16. >
  17. > Android Studio连接设备调试操作,可参考<https://developer.android.com/studio/run/device?hl=zh-cn>。
  18. >
  19. > 手机需开启“USB调试模式”,Android Studio 才能识别到手机。 华为手机一般在设置->系统和更新->开发人员选项->USB调试中开始“USB调试模型”。
  20. ![run_app](images/run_app.PNG)
  21. 3. 在Android设备上,点击“继续安装”,安装完即可查看到设备摄像头捕获的内容和推理结果。
  22. ![install](images/install.jpg)
  23. 如下图所示,识别出的概率最高的物体是植物。
  24. ![result](images/app_result.jpg)
  25. 4. Demo部署问题解决方案。
  26. 4.1 NDK、CMake、JDK等工具问题:
  27. 如果Android Studio内安装的工具出现无法识别等问题,可重新从相应官网下载和安装,并配置路径。
  28. - NDK >= 21.3 [NDK](https://developer.android.google.cn/ndk/downloads?hl=zh-cn)
  29. - CMake >= 3.10.2 [CMake](https://cmake.org/download)
  30. - Android SDK >= 26 [SDK](https://developer.microsoft.com/zh-cn/windows/downloads/windows-10-sdk/)
  31. - JDK >= 1.8 [JDK](https://www.oracle.com/cn/java/technologies/javase/javase-jdk8-downloads.html)
  32. ![project_structure](images/project_structure.png)
  33. 4.2 NDK版本不匹配问题:
  34. 打开`Android SDK`,点击`Show Package Details`,根据报错信息选择安装合适的NDK版本。
  35. ![NDK_version](images/NDK_version.jpg)
  36. 4.3 Android Studio版本问题:
  37. 在`工具栏-help-Checkout for Updates`中更新Android Studio版本。
  38. 4.4 Gradle下依赖项安装过慢问题:
  39. 如图所示, 打开Demo根目录下`build.gradle`文件,加入华为镜像源地址:`maven {url 'https://developer.huawei.com/repo/'}`,修改classpath为4.0.0,点击`sync`进行同步。下载完成后,将classpath版本复原,再次进行同步。
  40. ![maven](images/maven.jpg)
  41. ## 示例程序详细说明
  42. 风格Android示例程序通过Android Camera 2 API实现摄像头获取图像帧,以及相应的图像处理等功能,在[Runtime](https://www.mindspore.cn/tutorial/lite/zh-CN/master/use/runtime.html)中完成模型推理的过程。
  43. ### 示例程序结构
  44. ```text
  45. ├── app
  46. │   ├── build.gradle # 其他Android配置文件
  47. │   ├── download.gradle # APP构建时由gradle自动从HuaWei Server下载依赖的库文件及模型文件
  48. │   ├── proguard-rules.pro
  49. │   └── src
  50. │   ├── main
  51. │   │   ├── AndroidManifest.xml # Android配置文件
  52. │   │   ├── java # java层应用代码
  53. │   │   │   └── com
  54. │   │   │   └── mindspore
  55. │   │   │   └── posenetdemo # 图像处理及推理流程实现
  56. │   │   │   ├── CameraDataDealListener.java
  57. │   │   │   ├── ImageUtils.java
  58. │   │   │   ├── MainActivity.java
  59. │   │   │   ├── PoseNetFragment.java
  60. │   │   │   ├── Posenet.java #
  61. │   │   │   └── TestActivity.java
  62. │   │   └── res # 存放Android相关的资源文件
  63. │   └── test
  64. └── ...
  65. ```
  66. ### 下载及部署模型文件
  67. 从MindSpore Model Hub中下载模型文件,本示例程序中使用的目标检测模型文件为`style_predict_quant.ms`、`style_transfer_quant.ms`,同样通过`download.gradle`脚本在APP构建时自动下载,并放置在`app/src/main/assets`工程目录下。
  68. > 若下载失败请手动下载模型文件,style_predict_quant.ms [下载链接](https://download.mindspore.cn/model_zoo/official/lite/style_lite/style_predict_quant.ms),以及style_transfer_quant.ms [下载链接](https://download.mindspore.cn/model_zoo/official/lite/style_lite/style_transfer_quant.ms)。
  69. ### 编写端侧推理代码
  70. 在风格迁移demo中,使用Java API实现端测推理。相比于C++ API,Java API可以直接在Java Class中调用,无需实现JNI层的相关代码,具有更好的便捷性。
  71. 风格迁移demo推理代码流程如下,完整代码请参见:`src/main/java/com/mindspore/styletransferdemo/StyleTransferModelExecutor.java`。
  72. 1. 加载MindSpore Lite模型文件,构建上下文、会话以及用于推理的计算图。
  73. - 加载模型:从文件系统中读取MindSpore Lite模型,并进行模型解析。
  74. ```java
  75. // Load the .ms model.
  76. style_predict_model = new Model();
  77. if (!style_predict_model.loadModel(mContext, "style_predict_quant.ms")) {
  78. Log.e("MS_LITE", "Load style_predict_model failed");
  79. }
  80. style_transform_model = new Model();
  81. if (!style_transform_model.loadModel(mContext, "style_transfer_quant.ms")) {
  82. Log.e("MS_LITE", "Load style_transform_model failed");
  83. }
  84. ```
  85. - 创建配置上下文:创建配置上下文`MSConfig`,保存会话所需的一些基本配置参数,用于指导图编译和图执行。
  86. ```java
  87. msConfig = new MSConfig();
  88. if (!msConfig.init(DeviceType.DT_CPU, NUM_THREADS, CpuBindMode.MID_CPU)) {
  89. Log.e("MS_LITE", "Init context failed");
  90. }
  91. ```
  92. - 创建会话:创建`LiteSession`,并调用`init`方法将上一步得到`MSConfig`配置到会话中。
  93. ```java
  94. // Create the MindSpore lite session.
  95. Predict_session = new LiteSession();
  96. if (!Predict_session.init(msConfig)) {
  97. Log.e("MS_LITE", "Create Predict_session failed");
  98. msConfig.free();
  99. }
  100. Transform_session = new LiteSession();
  101. if (!Transform_session.init(msConfig)) {
  102. Log.e("MS_LITE", "Create Predict_session failed");
  103. msConfig.free();
  104. }
  105. msConfig.free();
  106. ```
  107. - 加载模型文件并构建用于推理的计算图
  108. ```java
  109. // Compile graph.
  110. if (!Predict_session.compileGraph(style_predict_model)) {
  111. Log.e("MS_LITE", "Compile style_predict graph failed");
  112. style_predict_model.freeBuffer();
  113. }
  114. if (!Transform_session.compileGraph(style_transform_model)) {
  115. Log.e("MS_LITE", "Compile style_transform graph failed");
  116. style_transform_model.freeBuffer();
  117. }
  118. // Note: when use model.freeBuffer(), the model can not be compile graph again.
  119. style_predict_model.freeBuffer();
  120. style_transform_model.freeBuffer();
  121. ```
  122. 2. 输入数据: Java目前支持`byte[]`或者`ByteBuffer`两种类型的数据,设置输入Tensor的数据。
  123. - 在输入数据之前,将float数组转换为byte数组。
  124. ```java
  125. public static byte[] floatArrayToByteArray(float[] floats) {
  126. ByteBuffer buffer = ByteBuffer.allocate(4 * floats.length);
  127. buffer.order(ByteOrder.nativeOrder());
  128. FloatBuffer floatBuffer = buffer.asFloatBuffer();
  129. floatBuffer.put(floats);
  130. return buffer.array();
  131. }
  132. ```
  133. - 通过`ByteBuffer`输入数据。`contentImage`为用户提供的图片,`styleBitmap`为预置风格图片。
  134. ```java
  135. public ModelExecutionResult execute(Bitmap contentImage, Bitmap styleBitmap) {
  136. Log.i(TAG, "running models");
  137. fullExecutionTime = SystemClock.uptimeMillis();
  138. preProcessTime = SystemClock.uptimeMillis();
  139. ByteBuffer contentArray =
  140. ImageUtils.bitmapToByteBuffer(contentImage, CONTENT_IMAGE_SIZE, CONTENT_IMAGE_SIZE, 0, 255);
  141. ByteBuffer input = ImageUtils.bitmapToByteBuffer(styleBitmap, STYLE_IMAGE_SIZE, STYLE_IMAGE_SIZE, 0, 255);
  142. ```
  143. 3. 对输入Tensor按照模型进行推理,获取输出Tensor,并进行后处理。
  144. - 使用`runGraph`对预置图片进行模型推理,并获取结果`Predict_results`。
  145. ```java
  146. List<MSTensor> Predict_inputs = Predict_session.getInputs();
  147. if (Predict_inputs.size() != 1) {
  148. return null;
  149. }
  150. MSTensor Predict_inTensor = Predict_inputs.get(0);
  151. Predict_inTensor.setData(input);
  152. preProcessTime = SystemClock.uptimeMillis() - preProcessTime;
  153. stylePredictTime = SystemClock.uptimeMillis();
  154. if (!Predict_session.runGraph()) {
  155. Log.e("MS_LITE", "Run Predict_graph failed");
  156. return null;
  157. }
  158. stylePredictTime = SystemClock.uptimeMillis() - stylePredictTime;
  159. Log.d(TAG, "Style Predict Time to run: " + stylePredictTime);
  160. // Get output tensor values.
  161. List<String> tensorNames = Predict_session.getOutputTensorNames();
  162. Map<String, MSTensor> outputs = Predict_session.getOutputMapByTensor();
  163. Set<Map.Entry<String, MSTensor>> entry = outputs.entrySet();
  164. float[] Predict_results = null;
  165. for (String tensorName : tensorNames) {
  166. MSTensor output = outputs.get(tensorName);
  167. if (output == null) {
  168. Log.e("MS_LITE", "Can not find Predict_session output " + tensorName);
  169. return null;
  170. }
  171. int type = output.getDataType();
  172. Predict_results = output.getFloatData();
  173. }
  174. ```
  175. - 利用上一步获取的结果,再次对用户图片进行模型推理,得到风格转换的数据`transform_results`。
  176. ```java
  177. List<MSTensor> Transform_inputs = Transform_session.getInputs();
  178. // transform model have 2 input tensor, tensor0: 1*1*1*100, tensor1;1*384*384*3
  179. MSTensor Transform_inputs_inTensor0 = Transform_inputs.get(0);
  180. Transform_inputs_inTensor0.setData(floatArrayToByteArray(Predict_results));
  181. MSTensor Transform_inputs_inTensor1 = Transform_inputs.get(1);
  182. Transform_inputs_inTensor1.setData(contentArray);
  183. styleTransferTime = SystemClock.uptimeMillis();
  184. if (!Transform_session.runGraph()) {
  185. Log.e("MS_LITE", "Run Transform_graph failed");
  186. return null;
  187. }
  188. styleTransferTime = SystemClock.uptimeMillis() - styleTransferTime;
  189. Log.d(TAG, "Style apply Time to run: " + styleTransferTime);
  190. postProcessTime = SystemClock.uptimeMillis();
  191. // Get output tensor values.
  192. List<String> Transform_tensorNames = Transform_session.getOutputTensorNames();
  193. Map<String, MSTensor> Transform_outputs = Transform_session.getOutputMapByTensor();
  194. float[] transform_results = null;
  195. for (String tensorName : Transform_tensorNames) {
  196. MSTensor output1 = Transform_outputs.get(tensorName);
  197. if (output1 == null) {
  198. Log.e("MS_LITE", "Can not find Transform_session output " + tensorName);
  199. return null;
  200. }
  201. transform_results = output1.getFloatData();
  202. }
  203. ```
  204. - 对输出节点的数据进行处理,得到推理后的最终结果。
  205. ```java
  206. float[][][][] outputImage = new float[1][][][]; // 1 384 384 3
  207. for (int x = 0; x < 1; x++) {
  208. float[][][] arrayThree = new float[CONTENT_IMAGE_SIZE][][];
  209. for (int y = 0; y < CONTENT_IMAGE_SIZE; y++) {
  210. float[][] arrayTwo = new float[CONTENT_IMAGE_SIZE][];
  211. for (int z = 0; z < CONTENT_IMAGE_SIZE; z++) {
  212. float[] arrayOne = new float[3];
  213. for (int i = 0; i < 3; i++) {
  214. int n = i + z * 3 + y * CONTENT_IMAGE_SIZE * 3 + x * CONTENT_IMAGE_SIZE * CONTENT_IMAGE_SIZE * 3;
  215. arrayOne[i] = transform_results[n];
  216. }
  217. arrayTwo[z] = arrayOne;
  218. }
  219. arrayThree[y] = arrayTwo;
  220. }
  221. outputImage[x] = arrayThree;
  222. }
  223. Bitmap styledImage =
  224. ImageUtils.convertArrayToBitmap(outputImage, CONTENT_IMAGE_SIZE, CONTENT_IMAGE_SIZE);
  225. postProcessTime = SystemClock.uptimeMillis() - postProcessTime;
  226. fullExecutionTime = SystemClock.uptimeMillis() - fullExecutionTime;
  227. Log.d(TAG, "Time to run everything: $" + fullExecutionTime);
  228. return new ModelExecutionResult(styledImage,
  229. preProcessTime,
  230. stylePredictTime,
  231. styleTransferTime,
  232. postProcessTime,
  233. fullExecutionTime,
  234. formatExecutionLog());
  235. ```