上一篇中我们分析了android的启动脚本之类的东西,在这一篇中我们将分析一下do_install 这个函数,由于这个脚本是包含在install.img文件中,我们先看一下这个包里有那些文件,在这里我以树的形式列出包含有那些文件:
install# tree . ├── bin │ ├── cfdisk │ ├── dialog │ └── pv ├── grub │ ├── android-x86.xpm.gz │ ├── e2fs_stage1_5 │ ├── fat_stage1_5 │ ├── iso9660_stage1_5 │ ├── ntfs_stage1_5 │ ├── stage1 │ ├── stage2 │ └── stage2_eltorito ├── lib │ ├── libncursesw.so.5 │ ├── libntfs.so.10 │ ├── libtinfo.so.5 │ ├── libuuid.so.1 │ └── terminfo │ └── l │ └── linux ├── sbin │ ├── grub │ └── mkntfs └── scripts └── 1-install
这个函数包含在1-install这个文件里,可以看一下这个函数的实现:
do_install() { until install_hd; do if [ $retval -eq 255 ]; then dialog --title ' Error! ' --yes-label Retry --no-label Reboot \ --yesno '\nInstallation failed! Please check if you have enough free disk space to install Android-x [ $? -eq 1 ] && rebooting fi done [ -n "$VESA" ] || runit="Run Android-x86" dialog --clear --title ' Congratulations! ' \ --menu "\n Android-x86 is installed successfully.\n " 11 51 13 \ "$runit" "" "Reboot" "" 2> $tempfile case "`cat $tempfile`" in Run*) cd /android umount system if mountpoint -q /sfs; then umount /sfs mount -o loop /hd/$asrc/system.sfs /sfs mount -o loop /sfs/system.img system else mount -o loop /hd/$asrc/system.img system fi if [ -d /hd/$asrc/data ]; then mount --bind /hd/$asrc/data data elif [ -e /hd/$asrc/data.img ]; then mount -o loop /hd/$asrc/data.img data fi ;; *) rebooting ;; esac
在这个函数中使用了until这样一个关键字,这个是shell脚本的语句,具体的作用大家google吧,接下来调用了install_hd这个函数,这个函数也是在这个文件里定义的,具体实现如下:
install_hd() { select_dev || rebooting retval=1 case "$choice" in [sh]d*) install_to $choice retval=$? ;; Create*) cfdisk ;; Detect*) dialog --title " Detecting... " --nocancel --pause "" 8 41 1 ;; esac return $retval }
在这个函数里可以看到调用了一个select_dev的函数,这个函数也定义在该文件中,定义如下:
select_dev() { fdisk -l | grep ^/dev | cut -b6-12,55- | awk '{ if (!match($2, "Extended")) { printf("\"%-28s", $0) system("echo -n `cat /sys/block/*/"$1"/../device/model`") printf("\" \"\"\n") } } END { printf("\"Create/Modify partitions\" \"\"\n\"Detect devices\" \"\"") }' > $menufile choose "Choose Partition" "Please select a partition to install Android-x86:" return $retval }
这个函数实现的就是下面的这张图:
在上面的函数中还调用了另外一个函数,choose,该函数定义如下:
choose() { dialog --clear --title " $1 " \ --menu "$2" 20 71 13 --file $menufile 2> $tempfile retval=$? choice=`cat $tempfile` }
在这个函数中需要注意的是最后一个choice值,这个值在后面会用到。接下来我们回到install_hd,这个函数,继续往下执行:
case "$choice" in
[sh]d*)
install_to $choice
retval=$?
;;
Create*)
cfdisk
;;
Detect*)
dialog --title " Detecting... " --nocancel --pause "" 8 41 1
;;
esac
因为此处我们主要讲解安装,所以其他的来那个选项我们暂时不分析了,等后面在补上。我们看一下上述表红色的部分。其中retval=$?用来取得调用select_dev函数的返回值。接下来我们看一下install_to这个函数,这个函数也是定义在这个文件里,由于这个函数内容比较多,我们一段一段分析:
cd / mountpoint -q /hd && umount /hd while [ 1 ]; do format_fs $1 try_mount rw /dev/$1 /hd && break dialog --clear --title " Error " --defaultno --yesno \ "\n Cannot mount /dev/$1\n Do you want to format it?" 8 37 [ $? -ne 0 ] && return 255 done
这一段代码的主要功能是完成对磁盘的分区与格式化,具体的调用是在format_fs中实现,该函数实现如下:
format_fs() { local cmd echo -e '"Do not format" ""\next3 ""\next2 ""\nntfs ""\nfat32 ""' > $menufile choose "Choose filesystem" "Please select a filesystem to format $1:" case "$choice" in ext3) cmd="mke2fs -jL" ;; ext2) cmd="mke2fs -L" ;; ntfs) cmd="mkntfs -fL" ;; fat32) cmd="newfs_msdos -L" ;; *) ;; esac if [ -n "$cmd" ]; then dialog --title " Confirm " --no-label Skip --yesno \ "\n You chose to format $1 to $choice.\n All data in that partition will LOSE.\n\n Are you sure to format the partition $1?" 10 51 [ $? -ne 0 ] && return 1 $cmd Android-x86 /dev/$1 | awk '{ # FIXME: very imprecise progress if (match($0, "done")) printf("%d\n", i+=33) }' | progress_bar "Formatting" "Formatting partition $1..." fi }
实现的界面对应如下:
fs=`cat /proc/mounts | grep /dev/$1 | awk '{ print $3 }'` asrc=android-$VER dialog --title " Confirm " --no-label Skip --defaultno --yesno \ "\n Do you want to install boot loader GRUB?" 7 47 if [ $? -eq 0 ]; then cp -af /grub /hd d=0 while [ 1 ]; do h=`echo $d | awk '{ printf("%c", $1+97) }'` [ -d /sys/block/[sh]d$h/$1 ] && break d=$(($d+1)) done p=$((`echo $1 | cut -b4-`-1)) create_menulst $p create_winitem $1 $d rm -f /hd/boot/grub/stage1 echo "setup (hd$d) (hd$d,$p)" | grub > /dev/tty5 [ $? -ne 0 ] && return 255 fi
以上的代码对应如下界面:
此处我们选择安装grub,接着执行下面的内容,
[ "$fs" != "vfat" -a "$fs" != "fuseblk" ] && dialog --title " Question " --yesno \ "\nDo you want to install /system directory as read-write?\n\nMaking /system be read-write is easier for debugging, but it needs more disk space and longer installation time." 10 61 instal_rw=$?
以上的内容将显示出如下界面:
此处我们选择YES,接着执行下面的内容,代码如下:
files="mnt/$SRC/kernel mnt/$SRC/initrd.img mnt/$SRC/ramdisk.img" [ $instal_rw -eq 0 ] && files="$files android/system" || files="$files mnt/$SRC/system.*" size=0 for s in `du -sk $files | awk '{print $1}'`; do size=$(($size+$s)) done mkdir hd/$asrc cd hd/$asrc rm -rf system* ( ( cd /; find $files | $CPIO -H newc -o ) | pv -ns ${size}k | ( $CPIO -iud > /dev/null; echo $? > /tmp/result )) 2>&1 \ | progress_bar "Installing Android-x86" "Expect to write $size KB..." result=$((`cat /tmp/result`*255)) if [ $result -eq 0 ]; then chmod 644 * chown 0.0 * mv mnt/$SRC/* . && rm -rf mnt [ -d android ] && mv android/* . && rmdir android case "$fs" in vfat|fuseblk) create_data_img ;; *) mkdir data ;; esac fi sync
上面的代码界面如下:
接下就根据函数的调用关系,我们跳转到此文开头的install函数,接着执行下面的内容,如果上面的东西全部安装完成,那最后则会调用以下代码,提示用户以安装完成
[ -n "$VESA" ] || runit="Run Android-x86" dialog --clear --title ' Congratulations! ' \ --menu "\n Android-x86 is installed successfully.\n " 11 51 13 \ "$runit" "" "Reboot" "" 2> $tempfile case "`cat $tempfile`" in Run*) cd /android umount system if mountpoint -q /sfs; then umount /sfs mount -o loop /hd/$asrc/system.sfs /sfs mount -o loop /sfs/system.img system else mount -o loop /hd/$asrc/system.img system fi if [ -d /hd/$asrc/data ]; then mount --bind /hd/$asrc/data data elif [ -e /hd/$asrc/data.img ]; then mount -o loop /hd/$asrc/data.img data fi ;; *) rebooting ;; esac
上面的代码实现的界面如下:
此时android的系统已经安装完成了,可以选择重启系统了。我们将会在下一篇讲解android的启动。