android-x86 install安装流程注解

#
# By Chih-Wei Huang <cwhuang@linux.org.tw>
#
# License: GNU Public License
# We explicitely grant the right to use the scripts
# with Android-x86 project.
#

tempfile=/tmp/temp-$$
menufile=/tmp/menu-$$

CPIO=cpio
OS_TITLE=${OS_TITLE:-Android-x86}

rebooting()
{
    dialog --title " Rebooting... " --nocancel --pause "" 8 41 1
    sync
    umount -a
    reboot -f
}

auto_answer()
{
    echo "$answer" > $tempfile
    unset answer
    test "`cat $tempfile`" != "0"
}

set_answer_if_auto()
{
    [ -n "$AUTO_INSTALL" ] && answer="$1"
}

adialog()
{
    if [ -n "$answer" ]; then
        auto_answer
    else
        #将用户的选择结果保存在tempfile中
        dialog "$@" 2> $tempfile
    fi
}

choose()
{
    #第1条语句是title,第2条语句是显示内容,menufile中的文本内容为选项
    adialog --clear --title " $1 " --menu "$2" 21 79 13 --file $menufile

    #上一命令正常执行,所以retval=0
    retval=$?
    #choice即为用户的选择结果
    choice=`cat $tempfile`
}

size_gb()
{
    printf %0.2fGB $(dc `cat $1/size` 2097152 / p)
}

find_partition()
{
    grep -H ^$2$ /sys/block/$1/*/partition 2> /dev/null | cut -d/ -f5
}

list_disks()
{
    for b in /sys/block/[shv]d[a-z] /sys/block/mmcblk? /sys/block/nvme*; do
        [ -d $b ] && echo $b
    done
}

auto_partition()
{
    [ "$AUTO_INSTALL" = "force" ] || dialog --title " Auto Installer " --defaultno --yesno \
        "\nYou have chosen the AUTO installation.\n\nThe installer will erase the whole /dev/$1 and install $OS_TITLE to it.\n\nThis is the last confirmation. Are you sure to do so?" 12 61
    [ $? -ne 0 ] && rebooting

    if [ -z "$efi" ]; then
        echo -e "o\nn\np\n1\n\n\nw\n" | fdisk /dev/$1
        p=1
    else
        sgdisk --zap-all /dev/$1
        sgdisk --new=1::+260M --typecode=1:EF00 --largest-new=2 --typecode=2:8300 /dev/$1
        p=2
    fi > /dev/tty6

    while sleep 1; do
        answer=`find_partition $1 $p`
        [ -n "$answer" ] && break
    done
    [ -n "$efi" ] && mkdosfs -n EFI /dev/`find_partition $1 1`
}

#使用cfdisk工具进行分区
partition_drive()
{
    #清空menufile文件
    echo -n > $menufile
    #list_disks函数用来列举出符合条件的块设备
    for i in `list_disks`; do
        #将basename添加到menufile中,比如sda
        echo -n `basename $i` >> $menufile
        #-f表示是文件,判断块设备sda(硬盘)是否可以被删除,如果$i/removable中的值为0,表示不可被删除
        if [ -f $i/removable -a `cat $i/removable` -eq 0 ]; then
            echo -n ' "Harddisk ' >> $menufile
        else
            echo -n ' "Removable' >> $menufile
        fi
        #块设备(硬盘)大小
        if [ -f $i/size ]; then
            sz=$(size_gb $i)
            [ "$sz" = "0.00GB" ] && sz="<0.01GB"
            #将硬盘大小追加到menufil
            printf " %10s" $sz >> $menufile
        fi
        #将硬盘品牌追加到menufile,比如:KXG6AZNV512G TOSHIBA    
        for f in $i/device/model $i/*/name; do
            [ -e $f ] && echo -n " `sed $'s/\x04//g' $f`" >> $menufile && break
        done
        #将" *"追加到menufile
        [ "`basename $i`" = "$booted_from" -o -d $i/$booted_from ] && echo -n " *" >> $menufile
        #将'"'追加到menufile,经过测试,最终menufile的内容为:sda "Harddisk 10.00GB QEMU HARDDISK "
        echo '"' >> $menufile
    done
    #wc -l用来显示文本行数,menufile只有一行,所以count=1
    count=`wc -l < $menufile`
    if [ $count -eq 0 ]; then
        dialog --title " Error " --msgbox \
            "\nOK. There is no hard drive to edit partitions." 8 49
        return 255
    fi
    #count=1,所以drive=1
    if [ $count -eq 1 -o "$AUTO_INSTALL" = "force" ]; then
        drive=1
    else
        drive=`basename $AUTO_INSTALL`
    fi
    #读出menufile中的第1个元素,也就是硬盘名,下面命令处理后的choice=sda
    choice=`awk -v n=$drive '{ if (n == NR || n == $1) print $1 }' $menufile`
    #/dev/sda是否块设备,结果:是
    if [ -b /dev/$choice ]; then
        retval=0
    else
        choose "Choose Drive" "Please select a drive to edit partitions:\n\n* - Installer source"
    fi
    #retval=0,下面判断成立
    if [ $retval -eq 0 ]; then
        #-n表示非空串,不成立
        if [ -n "$AUTO_INSTALL" ]; then
            auto_partition $choice
            return 1
        fi
        
        #fdisk -l列出所有分区表,最终结果是:part_tool=cfdisk
        if fdisk -l /dev/$choice | grep -q GPT; then
            part_tool=cgdisk
        elif fdisk -l /dev/$choice | grep -q doesn.t; then
            dialog --title " Confirm " --defaultno --yesno "\n Do you want to use GPT?" 7 29
            #选择了否,part_tool=cfdisk
            [ $? -eq 0 ] && part_tool=cgdisk || part_tool=cfdisk
        else
            part_tool=cfdisk
        fi
        
        #下面命令的完整形式是:cfdisk /dev/sda,使用cfdisk命令进行分区,进入创建分区界面
        $part_tool /dev/$choice
        #经过一系列的分区操作后,分区完成,retval=1
        if [ $? -eq 0 ]; then
            retval=1
        else
            retval=255
        fi
    fi
    return $retval
}

#选择分区
select_dev()
{
    #用blkid可列出当前系统中所有已挂载文件系统的类型,blkid主要用来对系统的块设备(包括交换分区)所使用的文件系统类型、LABEL、UUID等信息进行查询
    #举个例子,比如:/dev/sr0: LABEL="Android-x86 2021-11-12(x86-64)" TYPE="iso9660"
    #grep -v 去除掉/dev/block/ 和 /dev/loop
    #cut -b6- 每行从第6位开始,其实也就是将/dev/裁减掉
    #sort 以默认的方式将文本文件的第一列以 ASCII 码的次序排列
    #awk 处理文本,每行只保留TYPE和LABEL
    #将处理后的文本保存至/tmp/temp-当前进程的进程号
    #文本内容为:sr0 iso9660 Android-x86
    blkid | grep -v -E "^/dev/block/|^/dev/loop" | cut -b6- | sort | awk '{
        l=""
        t="unknown"
        sub(/:/, "", $1)
        for (i = NF; i > 1; --i)
            if (match($i, "^TYPE")) {
                t=$i
                gsub(/TYPE=|"/, "", t)
            } else if (match($i, "^LABEL")) {
                l=$i
                gsub(/LABEL=|"/, "", l)
            }
        printf("%-11s%-12s%-18s\n", $1, t, l)
    }' > $tempfile

    #当分区已经存在,下面命令会将分区信息打印显示出来
    #遍历/sys/block目录,查找出符合/sys/block/$d/$d*格式的文件夹,比如/sys/block/sda/sda1/:
    #再过滤掉文件名包含字段"loop|ram|sr|boot|rpmb"的文件:
    #读取/sys/block/$d/$d*/size大小,并打印
    for d in `ls /sys/block`; do
        for i in /sys/block/$d/$d*; do
            #-d表示是目录也就是文件夹,下面语句的意思是:如果是文件夹就继续向下执行
            [ -d $i ] || continue
            #如果文件名包含字段"loop|ram|sr|boot|rpmb"则将该文件过滤掉,也就是只保留文件名中不包含"loop|ram|sr|boot|rpmb"的文件
            echo $i | grep -qE "loop|ram|sr|boot|rpmb" && continue
            #如果tempfile不包含basename,则f为unknown
            f=$(grep "`basename $i`" $tempfile || printf "%-11s%-30s" `basename $i` unknown)
            #通过size_gb函数将/sys/block/$d/$d*/size中的值进行转化
            sz=$(size_gb $i)
            [ "$sz" = "0.00GB" ] || printf "$f%10s\n" $sz
        done
    done | awk -v b=$booted_from '{
        #如果分区已经完成,打印出分区名
        if (!match($1, b)) {
            printf("\"%s\" \"", $0)
            system("cd /sys/block/*/"$1"; for f in ../device/model ../device/name; do [ -e $f ] && printf %-17s \"`cat $f`\" && break; done")
            printf("\"\n")
        }
    } END {
        #将下面文字保存到menufile文件中,下面选项就是选择分区界面的item
        printf("\"\" \"\"\n\"Create/Modify partitions\" \"\"\n\"Detect devices\" \"\"")
    }' > $menufile
    
    #select_dev弹出框
    choose "Choose Partition" "Please select a partition to install $OS_TITLE:\n\nRecommended minimum free space - 4GB  |  Optimum free space >= 8GB\n\nPartition | Filesystem | Label            | Size     | Drive name/model"
    return $retval
}

progress_bar()
{
    dialog --clear --title " $1 " --gauge "\n $2" 8 70
}

convert_fs()
{
    if blkid /dev/$1 | grep -q ext2; then
        #tune2fs是调整和查看ext2/ext3文件系统的文件系统参数,Linux系统下面也有文件系统自检,而且是可以通过tune2fs命令,自行定义自检周期及方式。
        #-j:将ext2文件系统转换为ext3类型的文件系统,ext2可以转ext3 但不可以转回,转回数据丢失
        #添加日志功能,将ext2转换成ext3文件系统
        /system/bin/tune2fs -j /dev/$1
        #e2fsck命令用于检查使用 Linux ext2 档案系统的 partition 是否正常工作。
        e2fsck -fy /dev/$1
    fi

    #下面的判断成立
    if blkid /dev/$1 | grep -q ext3; then
        #转换成ext4
        /system/bin/tune2fs -O extents,uninit_bg /dev/$1
        e2fsck -fy /dev/$1
    fi
}

format_fs()
{
    local cmd
    #将要在"Choose filesystem 弹出框"中出现的4行item,将这4行item保存至menufile文件中
    echo -e '"Do not re-format" ""\next4 ""\nntfs ""\nfat32 ""' > $menufile
    set_answer_if_auto $FORCE_FORMAT
    #Choose filesystem 弹出框
    choose "Choose filesystem" "Please select a filesystem to format $1:"
    #选择了ext4
    case "$choice" in
        ext4)
            #走这一步
            cmd="mkfs.ext3 -L"
            ;;
        ntfs)
            cmd="mkntfs -fL"
            ;;
        fat32)
            cmd="mkdosfs -n"
            ;;
        *)
            ;;
    esac
    if [ -n "$cmd" ]; then
        #Confirm界面
        [ -n "$AUTO_INSTALL" ] || dialog --title " Confirm " --defaultno --yesno \
            "\n You chose to format $1 to $choice.\n All data in that partition will be LOST.\n\n Are you sure to format the partition $1?" 10 59
        #选择yes,继续向下执行,将/dev/sda1格式化成ext3文件系统,如果选择no直接返回
        [ $? -ne 0 ] && return 1
        #将硬盘/dev/sda1格式化成ext3文件系统
        $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..."
        #转换分区文件类型,将ext3转化成ext4
        convert_fs  $1
    elif blkid /dev/$1 | grep -q ext[23]; then
        dialog --clear --title " Warning " --yesno \
            "\nYou chose to install android-x86 to an ext2/3 filesystem. We suggest you convert it to ext4 for better reliability and performance." 9 62
        [ $? -eq 0 ] && convert_fs $1
    fi
}

create_entry()
{
    title=$1
    shift
    echo -e "title $title\n\tkernel /$asrc/kernel$vga $@ SRC=/$asrc\n\tinitrd /$asrc/initrd.img\n" >> $menulst
}

create_menulst()
{
    menulst=/hd/grub/menu.lst
    #经过测试,VESA为空串,此处vga未被赋值
    [ -n "$VESA" ] && vga=" vga=788 modeset=0"
    echo -e "${GRUB_OPTIONS:-default=0\ntimeout=6\nsplashimage=/grub/android-x86.xpm.gz\n}root (hd0,$1)\n" > $menulst

    create_entry "$OS_TITLE $VER" quiet $cmdline
    create_entry "$OS_TITLE $VER (Debug mode)" $cmdline DEBUG=2
    create_entry "$OS_TITLE $VER (Debug mode) gralloc.gbm" $cmdline DEBUG=2 GRALLOC=gbm
    create_entry "$OS_TITLE $VER (Debug mode) drmfb-composer gralloc.gbm" $cmdline DEBUG=2 HWC=drmfb GRALLOC=gbm
    create_entry "$OS_TITLE $VER (Debug mode) hwcomposer.drm gralloc.gbm" $cmdline DEBUG=2 HWC=drm GRALLOC=gbm
    create_entry "$OS_TITLE $VER (Debug mode) gralloc.minigbm" DEBUG=2 GRALLOC=minigbm
    create_entry "$OS_TITLE $VER (Debug mode) hwcomposer.drm_minigbm gralloc.minigbm" DEBUG=2 HWC=drm_minigbm GRALLOC=minigbm
    create_entry "$OS_TITLE $VER (Debug mode) hwcomposer.intel gralloc.intel" DEBUG=2 HWC=intel GRALLOC=intel
    create_entry "$OS_TITLE $VER (Debug nomodeset)" nomodeset $cmdline DEBUG=2
    create_entry "$OS_TITLE $VER (Debug video=LVDS-1:d)" video=LVDS-1:d $cmdline DEBUG=2
}

create_winitem()
{
    win=`fdisk -l /dev/$(echo $1 | cut -b-3) | grep ^/dev | cut -b6-12,55- | awk '{
        if (match($2, "NTFS"))
            print $1
    }' | head -1`
    #经过测试,win为空串
    if [ -n "$win" ]; then
        dialog --title " Confirm " --yesno \
            "\nThe installer found a Windows partition in /dev/$win.\n\nDo you want to create a boot item for Windows?" 9 59
        [ $? -ne 0 ] && return 1
        wp=$((`echo $win | cut -b4-`-1))
        echo -e "title Windows\n\trootnoverify (hd$d,$wp)\n\tchainloader +1\n" >> $menulst
    fi
}

check_data_img()
{
    losetup /dev/loop7 data.img
    if blkid /dev/loop7 | grep -q ext[23]; then
        dialog --clear --title " Warning " --yesno \
            "\nYour data.img is an ext2/3 filesystem. We suggest you convert it to ext4 for better reliability." 8 58
        [ $? -eq 0 ] && convert_fs loop7
    fi
    losetup -d /dev/loop7
}

gen_img()
{
    if [ "$fs" = "vfat" ]; then
        ( dd bs=1M count=$1 if=/dev/zero | pv -ns $1m | dd of=$2 ) 2>&1 \
            | progress_bar "Creating `basename $2`" "Expect to write $1 MB..."
    else
        dd if=/dev/zero bs=1 count=0 seek=$1M of=$2
    fi
}

create_img()
{
    bname=`basename $2`
    if [ -e $2 ]; then
        dialog --title " Confirm " --defaultno --yesno \
            "\n $bname exists. Overwrite it?" 7 38
        [ $? -ne 0 ] && return 255
        rm -f $2
    fi
    dialog --title " Question " --nook --nocancel --inputbox \
        "\nPlease input the size of the $bname in MB:" 8 63 $1 2> $tempfile
    size=`cat $tempfile`
    [ 0$size -le 0 ] && size=2048
    gen_img $size $2
}

create_data_img()
{
    dialog --title " Confirm " --yesno \
        "\nThe installer is going to create a disk image to save the user data. At least 2048MB free disk space is recommended.\n\nAre you sure to create the image?" 11 62

    if [ $? -eq 0 ]; then
        if create_img 2048 data.img; then
            losetup /dev/loop6 data.img
            mkfs.ext4 -L /data /dev/loop6 > /dev/tty6
        fi
        [ $? -ne 0 ] && dialog --msgbox "\n Failed to create data.img." 7 33
    else
        dialog --title " Warning " --msgbox \
            "\nOK. So data will be save to a RAMDISK(tmpfs), and lose after power off." 8 49
    fi
}

try_upgrade()
{
    [ -d $1 ] && return

    for d in hd/*; do
        [ -e "$d"/ramdisk.img -a -n "`ls "$d"/system* 2> /dev/null`" ] && echo \"`basename $d`\" \"\"
    done | sort -r > $menufile

    count=`wc -l < $menufile`
    if [ $count -gt 1 ]; then
        echo -e '"" ""\n"Install to new folder '`basename $1`'" ""' >> $menufile
        choose "Multiple older versions are found" "Please select one to upgrade:"
    elif [ $count -eq 1 ]; then
        eval choice=`awk '{ print $1 }' $menufile`
        set_answer_if_auto 1
        adialog --title " Question " --yesno \
            "\nAn older version $choice is detected.\nWould you like to upgrade it?" 8 61
        [ $? -eq 0 ] || choice=
    fi

    if [ -n "$choice" ]; then
        prev=hd/$choice
        if [ -d "$prev" ]; then
            mv $prev $1
            for d in `find hd -type l -maxdepth 1`; do
                [ "`readlink $d`" = "$choice" ] && ln -sf `basename $1` $d
            done
            rm -rf $1/data/dalvik-cache/* $1/data/system/wpa_supplicant
            [ -s $1/data/misc/wifi/wpa_supplicant.conf ] && sed -i 's/\(ctrl_interface=\)\(.*\)/\1wlan0/' $1/data/misc/wifi/wpa_supplicant.conf
        fi
    fi
}

get_part_info()
{
    d=0
    while [ 1 ]; do
        h=`echo $d | awk '{ printf("%c", $1+97) }'`
        for part in /sys/block/[shv]d$h/$1 /sys/block/mmcblk$d/$1 /sys/block/nvme0n$(($d+1))/$1; do
            [ -d $part ] && break 2
        done
        d=$(($d+1))
    done
    #part/partition文件中保存的值表示第几个分区,sda1是硬盘sda的第1个分区,所以p=1
    p=`cat $part/partition`
    #disk为sda
    disk=$(basename `dirname $part`)
}

wait_for_device()
{
    local t=`echo /sys/block/*/$1/partition`
    [ -f "$t" ] || return 1
    #由于/dev/sda1是块设备,不进入until循环体
    until [ -b /dev/$1 ]; do
        echo add > `dirname $t`/uevent
        sleep 1
    done
}

install_to()
{
    #$1=sda1
    #等待块设备,当/dev/$1是块设备时才继续往下执行
    wait_for_device $1 || return 1
    cd /
    #这一段代码的主要功能是完成对磁盘的格式化以及挂载,格式化是在format_fs中实现
    #注意:要使用一块新的硬盘,必须将它格式化建立合适的文件系统(linux:ext2,ext3等,windows:ntsf,fat32),并挂载到相应的目录下我们才可以使用。
    #使用mountpoint这条命令来确认某个目录是否”临时性“的被文件系统占用,如果被占用了,使用umount卸除目前挂在Linux目录中的文件系统
    mountpoint -q /hd && umount /hd
    #经过测试,$AUTO_UPDATE为空串,所以FORCE_FORMAT=ext4
    [ -n "$AUTO_UPDATE" ] && FORCE_FORMAT=no || FORCE_FORMAT=ext4
    while [ 1 ]; do
        #硬盘/dev/sda1的格式化
        format_fs $1
        #将dev/sda1挂载到/hd目录下,向/hd这个目录里写数据将会保存到硬盘dev/sda1里,rw表示可读可写
        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
        FORCE_FORMAT=ext4
    done

    #安装grub启动引导选项
    #经过测试,fs=ext4
    fs=`cat /proc/mounts | grep /dev/$1 | awk '{ print $3 }'`
    cmdline=`sed "s|\(initrd.*img\s*\)||; s|quiet\s*||; s|\(vga=\w\+\?\s*\)||; s|\(DPI=\w\+\?\s*\)||; s|\(AUTO_INSTALL=\w\+\?\s*\)||; s|\(INSTALL=\w\+\?\s*\)||; s|\(SRC=\S\+\?\s*\)||; s|\(DEBUG=\w\+\?\s*\)||; s|\(BOOT_IMAGE=\S\+\?\s*\)||; s|\(iso-scan/filename=\S\+\?\s*\)||; s|[[:space:]]*$||" /proc/cmdline`

    #由于INSTALL_PREFIX为空,所以asrc=android-$VER,经过测试,asrc=android-2021-11-12
    [ -n "$INSTALL_PREFIX" ] && asrc=$INSTALL_PREFIX || asrc=android-$VER
    set_answer_if_auto 1
    #efi为空串,弹出Confirm对话框
    [ -z "$efi" ] && adialog --title " Confirm " --no-label Skip --defaultno --yesno \
        "\n Do you want to install boot loader GRUB?" 7 47
    #选择yes
    if [ $? -eq 0 ]; then
        #获得分区编号,$p=1
        get_part_info $1
        #下面的判断不成立
        if fdisk -l /dev/$disk | grep -q GPT; then
            umount /hd
            dialog --title " Warning " --defaultno --yesno \
                "\nFound GPT on /dev/$disk. The legacy GRUB can't be installed to GPT. Do you want to convert it to MBR?\n\nWARNING: This is a dangerous operation. You should backup your data first." 11 63
            [ $? -eq 1 ] && rebooting
            plist=$(sgdisk --print /dev/$disk | awk '/^  / { printf "%s:", $1 }' | sed 's/:$//')
            sgdisk --gpttombr=$plist /dev/$disk > /dev/tty6
            until try_mount rw /dev/$1 /hd; do
                sleep 1
            done
        fi
        #将/grub文件夹复制到/hd目录下
        cp -af /grub /hd
        p=$(($p-1))
        #create_menulst 0;向menulst文件中填充属性值,grub目录下menu.lst定义启动选项
        create_menulst $p
        #create_winitem sda1 sda,向menulst文件中填充值
        create_winitem $1 $d
        rm -f /hd/boot/grub/stage1
        echo "(hd$d) /dev/$disk" > /hd/grub/device.map
        echo "setup (hd$d) (hd$d,$p)" | grub --device-map /hd/grub/device.map > /dev/tty5
        [ $? -ne 0 ] && return 255
    fi

    #由于efi为空串,所以下面的代码暂时可忽略
    [ -n "$efi" ] && adialog --title " Confirm " --no-label Skip --yesno \
        "\n Do you want to install EFI GRUB2?" 7 39
    if [ $? -eq 0 ]; then
        [ -z "$AUTO_INSTALL" -o -n "$AUTO_UPDATE" ] && for i in `list_disks`; do
            disk=`basename $i`
            esp=`sgdisk --print /dev/$disk 2> /dev/null | grep EF00 | awk '{print $1}'`
            [ -n "$esp" ] && boot=`find_partition $disk $esp` && break
        done
        if [ -z "$esp" ]; then
            get_part_info $1
            boot=$(blkid /dev/$disk* | grep -v $disk: | grep vfat | cut -d: -f1 | head -1)
            [ -z "$boot" ] && boot=`find_partition $disk 1` || boot=`basename $boot`
            esp=`cat /sys/block/$disk/$boot/partition`
        fi
        mkdir -p efi
        mountpoint -q efi && umount efi
        wait_for_device $boot
        until try_mount rw /dev/$boot efi; do
            dialog --title " Confirm " --defaultno --yesno "\n Cannot mount /dev/$boot.\n Do you want to format it?" 8 37
            [ $? -eq 0 ] && mkdosfs -n EFI /dev/$boot
        done
        if [ "$efi" = "32" ]; then
            grubcfg=efi/boot/grub/i386-efi/grub.cfg
            bootefi=bootia32.efi
        else
            grubcfg=efi/boot/grub/x86_64-efi/grub.cfg
            bootefi=BOOTx64.EFI
        fi
        if [ -d efi/efi/boot -a ! -s efi/efi/boot/android.cfg ]; then
            efidir=/efi/Android
        else
            efidir=/efi/boot
            rm -rf efi/efi/Android
        fi
        mkdir -p `dirname $grubcfg` efi$efidir
        cp -af grub2/efi/boot/* efi$efidir
        sed -i "s|VER|$VER|; s|CMDLINE|$cmdline|; s|OS_TITLE|$OS_TITLE|" efi$efidir/android.cfg
        [ -s efi/boot/grub/grubenv ] || ( printf %-1024s "# GRUB Environment Block%" | sed 's/k%/k\n/; s/   /###/g' > efi/boot/grub/grubenv )

        echo -e 'set timeout=5\nset debug_mode="(DEBUG mode)"' > $grubcfg
        # Our grub-efi doesn't support ntfs directly.
        # Copy boot files to ESP so grub-efi could read them
        if [ "$fs" = "fuseblk" ]; then
            cp -f src/kernel src/initrd.img efi$efidir
            echo -e "set kdir=$efidir\nset src=SRC=/$asrc" >> $grubcfg
        else
            echo -e "set kdir=/$asrc" >> $grubcfg
        fi
        echo -e '\nsource $cmdpath/android.cfg' >> $grubcfg
        if [ -d src/boot/grub/theme ]; then
            cp -R src/boot/grub/[ft]* efi/boot/grub
            find efi/boot/grub -name TRANS.TBL -delete
        fi

        # Checking for old EFI entries, removing them and adding new depending on bitness
        efibootmgr | grep -Eo ".{0,6}Android-x86" | cut -c1-4 > /tmp/efientries
        if [ -s /tmp/efientries ]; then
            set_answer_if_auto 1
            adialog --title " Question " --yesno "\nEFI boot entries for previous Android-x86 installations were found.\n\nDo you wish to delete them?" 10 61
            [ $? -eq 0 ] && while read entry; do efibootmgr -Bb "$entry" > /dev/tty4 2>&1; done < /tmp/efientries
        fi
        efibootmgr -v -c -d /dev/$disk -p $esp -L "Android-x86 $VER" -l $efidir/$bootefi > /dev/tty4 2>&1

        if [ -s efi/startup.nsh ]; then
            sed -i "s|\\\\efi\\\\Android|$efidir|; s|/|\\\\|g" efi/startup.nsh
        else
            echo $efidir/$bootefi | sed 's|/|\\|g' > efi/startup.nsh
        fi
    fi

    #升级更新,暂时 忽略
    try_upgrade hd/$asrc

    ! test -f hd/$asrc/system.img -o -d hd/$asrc/system
    set_answer_if_auto $?
    #弹出Question对话框
    adialog --title " Question " --defaultno --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
    #下面的判断成立
    if [ $? -eq 0 -a -e /sfs/system.img ]; then
        sysimg="/sfs/system.img"
    else
        sysimg="mnt/$SRC/system.*"
    fi
    #把镜像文件写入到磁盘,将files中的所有文件写到android9.qcow2
    #经过测试,files="mnt///kernel mnt///initrd.img mnt///ramdisk.img /sfs/system.img"
    files="mnt/$SRC/kernel mnt/$SRC/initrd.img mnt/$SRC/$RAMDISK $sysimg"
    size=0
    for s in `du -sk $files | awk '{print $1}'`; do
        size=$(($size+$s))
    done

    #创建文件hd/android-2021-11-12并进入
    mkdir -p hd/$asrc
    cd hd/$asrc
    #删除system*文件
    rm -rf system*
    #开始写入,使用cpio解压img文件
    ( ( cd /; find $files | $CPIO -H newc -o ) | pv -ns ${size}k | ( $CPIO -iud > /dev/null; echo $? > /tmp/result )) 2>&1 \
        | progress_bar "Installing $OS_TITLE to $1" "Expect to write $size KB..."
    #经过测试result=0
    result=$((`cat /tmp/result`*255))

    #安装成功
    if [ $result -eq 0 ]; then
        for d in android mnt sfs ./$SRC; do
            [ -d $d ] && mv $d/* . && rmdir $d
        done
        chown 0.0 *
        for f in *; do
            [ -d $f ] || chmod 644 $f
        done

        #fs=ext4
        case "$fs" in
            vfat|fuseblk)
                [ -e data.img ] && check_data_img || create_data_img
                ;;
            *)
                mkdir -p data
                ;;
        esac
    fi

    dialog --infobox "\n Syncing to disk..." 5 27
    sync
    cd /

    return $result
}

#选择分区,创建分区,安装
install_hd()
{
    #经过测试,此处AUTO_INSTALL为空
    case "$AUTO_INSTALL" in
        [Uu]*)
            answer=${AUTO_UPDATE:-$(blkid | grep -v loop | grep -v iso9660 | sort | grep Android-x86 | cut -d: -f1 | head -1)}
            answer=${answer:-$(blkid | grep -v loop | sort | grep ext4 | cut -d: -f1 | head -1)}
            [ -b "$answer" -o -b /dev/$answer ] && answer=`basename $answer` || answer=
            AUTO_UPDATE=${answer:-$AUTO_UPDATE}
            [ -z "$AUTO_UPDATE" ] && AUTO_INSTALL=
            ;;
        *)
            #-z表示空串,[ -z "$answer" ]成立,执行函数set_answer_if_auto时由于AUTO_INSTALL为空,所以answer在这里没有被赋值
            [ -z "$answer" ] && set_answer_if_auto Create
            ;;
    esac

    #select_dev选择分区,选择分区失败会执行rebooting重新启动
    select_dev || rebooting
    retval=1
    #根据select_dev界面不同的选择,执行相应的代码
    case "$choice" in
        #创建分区
        Create*)
            #partition_drive创建分区
            partition_drive
            retval=$?
            ;;
        Detect*)
            dialog --title " Detecting... " --nocancel --pause "" 8 41 1
            ;;
        *)
            #安装到分区,假设$choice=sda1
            install_to $choice
            retval=$?
            ;;
    esac
    #如果$choice=Create*,也就是创建分区,那retval=1,这个值传递给do_install()函数中的until语句,表示继续循环,直到choice=0循环才终止,也就是要调用install_to()函数进行安装;
    #当partition_drive完成创建分区之后,此时在select_dev界面会出现3个选项,第1个选项就是分区项,当点击该项时,会执行install_to()函数进行安装
    return $retval
}

#安装流程的入口函数
do_install()
{
    #‘basename $dev’为系统命令,意思是:取文件路径的最后/后面的内容,比如:这里dev=/dev/block/sr0,booted_from=sr0 ,booted_from变量被引用时才执行该命令
    booted_from=`basename $dev`
    #2> /dev/null的意思是将标准错误stderr删掉,经过测试,此处efi为空
    efi=$(cat /sys/firmware/efi/fw_platform_size 2> /dev/null)
    #-n 表示非空串,此处efi为空,不满足
    [ -n "$efi" ] && mount -t efivarfs none /sys/firmware/efi/efivars

    #调用install_hd,主要是选择分区,创建分区,以及执行安装
    #当install_hd成立时,也就是install_hd返回0时,才退出循环
    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 $OS_TITLE." 8 51
            [ $? -eq 1 ] && rebooting
        fi
    done

    #如果上面的东西全部安装完成,那最后则会调用以下代码,提示用户以安装完成
    [ -n "$VESA" ] || runit="Run $OS_TITLE"
    dialog --clear --title ' Congratulations! ' \
        --menu "\n $OS_TITLE is installed successfully.\n " 11 51 13 \
        "$runit" "" "Reboot" "" 2> $tempfile
    case "`cat $tempfile`" in
        Run*)
            #Run Android-x86
            cd /android
            umount system
            mountpoint -q /sfs && umount /sfs
            if [ -e /hd/$asrc/system.sfs ]; then
                mount -o loop /hd/$asrc/system.sfs /sfs
                mount -o loop /sfs/system.img system
            elif [ -e /hd/$asrc/system.img ]; then
                mount -o loop /hd/$asrc/system.img system
            else
                mount --bind /hd/$asrc/system 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
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

renshuguo123723

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值