上一篇中我们分析了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的启动。