本文介绍的是Android系统OTA升级,主要介绍升级包的制作和Android代码中如何实现系统的升级。文章中的升级功能是在Firefly-RK3399板子在Android7.1.2上完成测试,其中下文中的增量升级未测试成功,文章中把该种方式记录下来用于方便以后再来研究。
目录
- 全量包
- 增量包
- Android代码调用升级
1. 全量包
命令
. build/envsetup.sh
lunch //然后选择你需要的配置(如15)
make -j8
make otapackage -j8
执行完上面的命令会在out/target/product/rk3399_firfly_box/rk3399_firefly_box-ota-en.jb.zip
的zip压缩包。其中/rk3399_firfly_box/
并非是固定的文件,这个文件夹大多是以自己板子的型号和厂商命名rk3399_firefly_box-ota-en.jb.zip
也是更具自己厂商和设备型号命名,把该压缩包命名成update.zip
也就完成了全量升级包的制作。
2. 增量包
Android增量的制作需要上一个旧的源码包和新的的源码包,其时我们通过制作全量包的命令就已经制作了源码包只是他们的所在的目录不同,
源码包路径
out/target/product/rk3399_firefly_box/obj/PACKAGING/target_files_intermediates/rk3399_firefly_box-target_files-eng.jb.zip
全量包路径
out/target/product/rk3399_firefly_box/rk3399_firefly_box-ota-eng.jb.zip
通过上面的两个路径我们可以知道,源码包和全量包生成的命名是相同的,只不过他们的存放路径不相同。其实他们的大小也不相同源码包要比全量包大(我测试中源码包为1.2G,全量包为567M),系统升级一般是用全量包和增量包升级。
制作增量包
制作增量包我们首先要准备一个最新版本的源码包和上一个版本的源码包如old.Zip和new.zip。然后我们在根目录创建创建一个OTA/
文件夹,分别为将olde.zip和new.zip复制到该文件夹中,然后执行下面命令
./build/tools/releasetools/ota_from_target_files -i ~/OTA/old.zip ~/OTA/new.zip ~/OTA/update.zip
上面的命令
-
./build/tools/releasetools/ota_from_target_files
为谷歌为我们提供的一个用于生产增量包的脚本。 -
-i
后面分别为旧版本源码包
和新版本源码包
的文件路径,做好一个是生产的增量包
路径。
执行上面的命令我们就会在根目录下的OTA文件夹下生产了一个update.zip
增量包
3. Android代码调用升级
如果您正好使用的是Firefly开发板供应商提供的板子,下面的这种方式可以不用看了。Firefly团队提供了一套方便开发者快速实现部分功能的api,该api中正好有系统升级功能的实现方法。请移步FireflyApi查看 安装升级
完成上面全量升级包
或增量升级包
都会得到update.zip
升级包。我们就可进行系统升级了,一般系统升级分为在线升级和离线升级。
在线升级需要将我们生产的全量包或者增量包放到服务器上,根据自己业务将其下载下来行升级。
离线升级需要我们将升级包从U盘中拷贝到Android系统中进行升级。
不管是在线升级还是离线升级我们都需要将升级包拷贝到Android外部储存的
根目录
中才能升级
本文用的是离线升级,是将升级包拷贝到Android外部储存的根目录中的,文章末尾的参考连接中有关于网络下载升级的介绍。
权限
<manifest ...
android:sharedUserId="android.uid.system" //系签名要添加的标识
...>
<uses-permission android:name="android.permission.RECOVERY" />
<uses-permission android:name="android.permission.REBOOT" />
</manifest>
因为上面两个权限需要将应用做出系统及app才可以使用,因此需要系统签名。文章的末尾的参考连接中有关于系统签名的实现方式。
获取升级包
/**
* 获取升级包
*
* @return 升级包file
*/
public File getUdapteFile() {
File updateFile = null;
if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
String updatePath = Environment.getExternalStorageDirectory().getAbsolutePath()
+ File.separator + "update.zip";
updateFile = new File(updatePath);
boolean isExists = updateFile.exists();
LogHelper.getInstance().d("是否存在升级包:"+isExists);
if (isExists) {
return updateFile;
}
}
return updateFile;
}
校验和升级
/**
* 系统升级
*/
public void systemUpdate() {
File udapterFile = getUdapteFile();
try {
//签名校验
RecoverySystem.verifyPackage(udapterFile, new RecoverySystem.ProgressListener() {
@Override
public void onProgress(int progress) {
LogHelper.getInstance().d("签名校验进度:" + progress);
}
}, null);
//升级
RecoverySystem.installPackage(this, udapterFile);
} catch (GeneralSecurityException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
}
上面的RecoverySystem.verifyPackage()
方法用于校验升级包的签名是否合法,我们在做升级名的时候make otapackage
命令执行完已经为我们签好了名,因此不需要在签名了。