1. 前言:
刷机,似乎是安卓手机用户的一项专利,但是,会刷机的用户一般都是喜新厌旧的角色,
一个系统用久了,就想换到另一个系统,或者觉得没有原来的好,或者又觉得要换回去,这样又要重刷。
但是刷来刷去都麻烦啊,并且每次刷机也不是没有风险的,一不小心就可能造成重要数据的丢失。
没有解决办法吗?有!双系统!甚至三系统,四系统!!
本文就是解决这个问题的,并且用本文中的方法,完全可以实现一键安装,一键卸载系统的功能,把系统的安装和卸载变成apk的安装和卸载一样简单。
(说明下,下面的方法以三星i93xx系列的手机为例的)
2. 先来简单介绍下安卓系统的启动过程:
在手机上电时,最先执行的集成到CPU芯片上的一段rom里的程序
这段程序负责加载nand flash或者sd卡上的引导程序,引导程序一般来讲都是uboot
uboot会完成一些设备的初始化,这里很重要的部分就是nand flash,以便将linux内核加载读到内存里并运行。
加载的内核根据情况,有可能是boot分区里的内核,也有可能是recovery分区里的内核。
内核跑起来之后,首先会挂载ramdisk到"/"根目录,然后执行/init创建第一个进程
init读取/init.rc,进行进一步的初始化,完成如创建目录,设置权限,挂载data,system,cache分区,启动一系列的service,包括重要的zygote进程
zygote进程又会创建system_server进程以完成进一步的初始化工作,并加载一系列的apk进程。
3. 接下来看下,一个安卓系统所需的分区:
一个uboot分区,负责引导内核
一个内核分区
一个system分区,用于存放安卓的系统程序和文件
一个data分区,用于存放系统的数据,apk程序,以及apk程序的数据等等。
一个cache分区,一般用于升级之用,用于保存ota升级包,升级日志等等。
4. 再按下来看下双系统的实现方案:
4.1 内核的引导问题
这是一个比较头痛的问题,上面讲到,内核是由uboot通过boot分区或者recovery分区加载进来的
如果还要加载其它分区的内核,就要考虑修改uboot的配置参数或者代码了
uboot的代码我们肯定是没有 的,
虽然一般来讲uboot的配置参数往往也是保存在某个分区里的,但一般都是加密的,所以也改不了。
所以我们只能考虑利用己有的分区了。
最简单的方法就是覆盖boot分区,将第二个安卓的boot.img写到boot分区,
然后写一个apk,当要启动哪个系统时,就把哪个系统的boot.img写到boot分区。
这种方式的缺点是切换麻烦,每次切换都要先启动其中的一个系统,然后运行apk进行切换。
然后,我们把贪婪的目光瞄向了recovery分区。
大家知道,recovery分区一般在系统升级或者恢复出厂设置的时候才会用到,所以我们考虑对recovery分区进行下手。
最简单的方法是把第二个安卓系统boot.img放到recovery分区里,这样可以实现触发进recovery来引导第二个安卓系统了。
当然,也可以在recovery分区里再放一个定制的uboot,从而实现更加灵活的加载方式,如可以显示引导菜单等等,再如从SD卡里加载内核等等。
但是这还是要有uboot的源码才行。
当然,在使用recovery分区之前,要对recovery分区作下备份。
使用recovery分区作为第二个系统的linux引导分区还有个好处就是一般手机都有开机进recovery的快捷键,
如三星的手机一般是在开机时同时按下:音量加,HOME,POWER三个按键就可以进recovery.
从而实现方便的系统切换。
4.2 system,data分区的创建问题:
解决了引导的问题,再来看下system和data分区的创建问题。
因为cache分区只在升级的时候会用到,所以两个系统可以共用,不用再创建了。
1)重新分区法:
也就是为每个系统建立不同的分区,这种方法需要对存储空间进行重新划分,显然比较麻烦,风险也比较高,可行性比较低。
2)使用虚拟磁盘的方案
大家一定对ubuntu能够在windows下直接安装的方式印象十分深刻
实际上ubuntu能够在不重新分区的情况下实现安装真是利用的虚拟磁盘实现的。
相对于第一种方案,避免了重新分区的麻烦。
虚拟磁盘是linux下很早内核就已经支持了,是很成熟的技术了。
所以这里虚拟磁盘是最好的选择,并且借助于虚拟磁盘,我们不仅可以实现双系统,还可以实现三系统,四系统,这完全取决于存储空间。
5. 理论讲清楚了,接下来,看下具体如何干吧。
首先,我们要创建一个system虚拟磁盘,这里有两种方法:
一种是从img直接生成虚拟磁盘,另一种方法是要将一个ota升级包中的system分区写到虚拟磁盘中。
第一种方法比较简单,只要执行一条命令即可:
simg2img system.img system.disk