mirror of https://github.com/ventoy/Ventoy.git
509 lines
12 KiB
Bash
509 lines
12 KiB
Bash
#!/bin/sh
|
|
|
|
#Ventoy partition 32MB
|
|
VENTOY_PART_SIZE=33554432
|
|
VENTOY_PART_SIZE_MB=32
|
|
VENTOY_SECTOR_SIZE=512
|
|
VENTOY_SECTOR_NUM=65536
|
|
|
|
ventoy_false() {
|
|
[ "1" = "2" ]
|
|
}
|
|
|
|
ventoy_true() {
|
|
[ "1" = "1" ]
|
|
}
|
|
|
|
|
|
vtinfo() {
|
|
echo -e "\033[32m$*\033[0m"
|
|
}
|
|
|
|
vtwarn() {
|
|
echo -e "\033[33m$*\033[0m"
|
|
}
|
|
|
|
|
|
vterr() {
|
|
echo -e "\033[31m$*\033[0m"
|
|
}
|
|
|
|
vtdebug() {
|
|
echo "$*" >> ./log.txt
|
|
}
|
|
|
|
vtoy_gen_uuid() {
|
|
if uuid -F BIN > /dev/null 2>&1; then
|
|
uuid -F BIN
|
|
elif uuidgen -V > /dev/null 2>&1; then
|
|
a=$(uuidgen | sed 's/-//g')
|
|
echo -en "\x${a:0:2}\x${a:2:2}\x${a:4:2}\x${a:6:2}\x${a:8:2}\x${a:10:2}\x${a:12:2}\x${a:14:2}\x${a:16:2}\x${a:18:2}\x${a:20:2}\x${a:22:2}\x${a:24:2}\x${a:26:2}\x${a:28:2}\x${a:30:2}"
|
|
elif python -V > /dev/null 2>&1; then
|
|
a=$(python -c 'import sys,uuid; sys.stdout.write(uuid.uuid4().hex)')
|
|
echo -en "\x${a:0:2}\x${a:2:2}\x${a:4:2}\x${a:6:2}\x${a:8:2}\x${a:10:2}\x${a:12:2}\x${a:14:2}\x${a:16:2}\x${a:18:2}\x${a:20:2}\x${a:22:2}\x${a:24:2}\x${a:26:2}\x${a:28:2}\x${a:30:2}"
|
|
elif [ -e /dev/urandom ]; then
|
|
dd if=/dev/urandom bs=1 count=16 status=none
|
|
else
|
|
datestr=$(date +%N%N%N%N%N)
|
|
a=${datestr:0:32}
|
|
echo -en "\x${a:0:2}\x${a:2:2}\x${a:4:2}\x${a:6:2}\x${a:8:2}\x${a:10:2}\x${a:12:2}\x${a:14:2}\x${a:16:2}\x${a:18:2}\x${a:20:2}\x${a:22:2}\x${a:24:2}\x${a:26:2}\x${a:28:2}\x${a:30:2}"
|
|
fi
|
|
}
|
|
|
|
check_tool_work_ok() {
|
|
|
|
if echo 1 | hexdump > /dev/null; then
|
|
vtdebug "hexdump test ok ..."
|
|
else
|
|
vtdebug "hexdump test fail ..."
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
if mkexfatfs -V > /dev/null; then
|
|
vtdebug "mkexfatfs test ok ..."
|
|
else
|
|
vtdebug "mkexfatfs test fail ..."
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
if vtoycli fat -T; then
|
|
vtdebug "vtoycli fat test ok ..."
|
|
else
|
|
vtdebug "vtoycli fat test fail ..."
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
vtdebug "tool check success ..."
|
|
ventoy_true
|
|
}
|
|
|
|
|
|
get_disk_part_name() {
|
|
DISK=$1
|
|
|
|
if echo $DISK | grep -q "/dev/loop"; then
|
|
echo ${DISK}p${2}
|
|
elif echo $DISK | grep -q "/dev/nvme[0-9][0-9]*n[0-9]"; then
|
|
echo ${DISK}p${2}
|
|
elif echo $DISK | grep -q "/dev/mmcblk[0-9]"; then
|
|
echo ${DISK}p${2}
|
|
elif echo $DISK | grep -q "/dev/nbd[0-9]"; then
|
|
echo ${DISK}p${2}
|
|
else
|
|
echo ${DISK}${2}
|
|
fi
|
|
}
|
|
|
|
check_umount_disk() {
|
|
DiskOrPart="$1"
|
|
grep "^${DiskOrPart}" /proc/mounts | while read mtline; do
|
|
mtpnt=$(echo $mtline | awk '{print $2}')
|
|
vtdebug "Trying to umount $mtpnt ..."
|
|
umount $mtpnt >/dev/null 2>&1
|
|
done
|
|
}
|
|
|
|
get_ventoy_version_from_cfg() {
|
|
if grep -q 'set.*VENTOY_VERSION=' $1; then
|
|
grep 'set.*VENTOY_VERSION=' $1 | awk -F'"' '{print $2}'
|
|
else
|
|
echo 'none'
|
|
fi
|
|
}
|
|
|
|
is_disk_contains_ventoy() {
|
|
DISK=$1
|
|
|
|
PART1=$(get_disk_part_name $1 1)
|
|
PART2=$(get_disk_part_name $1 2)
|
|
|
|
if [ -e /sys/class/block/${PART2#/dev/}/size ]; then
|
|
SIZE=$(cat /sys/class/block/${PART2#/dev/}/size)
|
|
else
|
|
SIZE=0
|
|
fi
|
|
|
|
if ! [ -b $PART1 ]; then
|
|
vtdebug "$PART1 not exist"
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
if ! [ -b $PART2 ]; then
|
|
vtdebug "$PART2 not exist"
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
PART1_TYPE=$(dd if=$DISK bs=1 count=1 skip=450 status=none | hexdump -n1 -e '1/1 "%02X"')
|
|
PART2_TYPE=$(dd if=$DISK bs=1 count=1 skip=466 status=none | hexdump -n1 -e '1/1 "%02X"')
|
|
|
|
# if [ "$PART1_TYPE" != "EE" ]; then
|
|
# if [ "$PART2_TYPE" != "EF" ]; then
|
|
# vtdebug "part2 type is $PART2_TYPE not EF"
|
|
# ventoy_false
|
|
# return
|
|
# fi
|
|
# fi
|
|
|
|
# PART1_TYPE=$(dd if=$DISK bs=1 count=1 skip=450 status=none | hexdump -n1 -e '1/1 "%02X"')
|
|
# if [ "$PART1_TYPE" != "07" ]; then
|
|
# vtdebug "part1 type is $PART2_TYPE not 07"
|
|
# ventoy_false
|
|
# return
|
|
# fi
|
|
|
|
if [ -e /sys/class/block/${PART1#/dev/}/start ]; then
|
|
PART1_START=$(cat /sys/class/block/${PART1#/dev/}/start)
|
|
fi
|
|
|
|
if [ "$PART1_START" != "2048" ]; then
|
|
vtdebug "part1 start is $PART1_START not 2048"
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
if [ "$VENTOY_SECTOR_NUM" != "$SIZE" ]; then
|
|
vtdebug "part2 size is $SIZE not $VENTOY_SECTOR_NUM"
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
ventoy_true
|
|
}
|
|
|
|
check_disk_secure_boot() {
|
|
if ! is_disk_contains_ventoy $1; then
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
PART2=$(get_disk_part_name $1 2)
|
|
|
|
vtoycli fat -s $PART2
|
|
}
|
|
|
|
get_disk_ventoy_version() {
|
|
|
|
if ! is_disk_contains_ventoy $1; then
|
|
ventoy_false
|
|
return
|
|
fi
|
|
|
|
PART2=$(get_disk_part_name $1 2)
|
|
|
|
ParseVer=$(vtoycli fat $PART2)
|
|
if [ $? -eq 0 ]; then
|
|
vtdebug "Ventoy version in $PART2 is $ParseVer"
|
|
echo $ParseVer
|
|
ventoy_true
|
|
return
|
|
fi
|
|
|
|
ventoy_false
|
|
}
|
|
|
|
wait_and_create_part() {
|
|
vPART1=$1
|
|
vPART2=$2
|
|
echo 'Wait for partitions ...'
|
|
for i in 0 1 2 3 4 5 6 7 8 9; do
|
|
if ls -l $vPART1 2>/dev/null | grep -q '^b'; then
|
|
if ls -l $vPART2 2>/dev/null | grep -q '^b'; then
|
|
break
|
|
fi
|
|
else
|
|
echo "Wait for $vPART1/$vPART2 ..."
|
|
sleep 1
|
|
fi
|
|
done
|
|
|
|
if ls -l $vPART1 2>/dev/null | grep -q '^b'; then
|
|
echo "$vPART1 exist OK"
|
|
else
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${vPART1#/dev/}/dev)
|
|
echo "mknod -m 0660 $vPART1 b $MajorMinor ..."
|
|
mknod -m 0660 $vPART1 b $MajorMinor
|
|
fi
|
|
|
|
if ls -l $vPART2 2>/dev/null | grep -q '^b'; then
|
|
echo "$vPART2 exist OK"
|
|
else
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${vPART2#/dev/}/dev)
|
|
echo "mknod -m 0660 $vPART2 b $MajorMinor ..."
|
|
mknod -m 0660 $vPART2 b $MajorMinor
|
|
fi
|
|
|
|
if ls -l $vPART1 2>/dev/null | grep -q '^b'; then
|
|
if ls -l $vPART2 2>/dev/null | grep -q '^b'; then
|
|
echo "partition exist OK"
|
|
fi
|
|
else
|
|
echo "[FAIL] $vPART1/$vPART2 does not exist"
|
|
exit 1
|
|
fi
|
|
}
|
|
|
|
|
|
format_ventoy_disk_mbr() {
|
|
reserve_mb=$1
|
|
DISK=$2
|
|
PARTTOOL=$3
|
|
|
|
PART1=$(get_disk_part_name $DISK 1)
|
|
PART2=$(get_disk_part_name $DISK 2)
|
|
|
|
sector_num=$(cat /sys/block/${DISK#/dev/}/size)
|
|
|
|
part1_start_sector=2048
|
|
|
|
if [ $reserve_mb -gt 0 ]; then
|
|
reserve_sector_num=$(expr $reserve_mb \* 2048)
|
|
part1_end_sector=$(expr $sector_num - $reserve_sector_num - $VENTOY_SECTOR_NUM - 1)
|
|
else
|
|
part1_end_sector=$(expr $sector_num - $VENTOY_SECTOR_NUM - 1)
|
|
fi
|
|
|
|
part2_start_sector=$(expr $part1_end_sector + 1)
|
|
|
|
modsector=$(expr $part2_start_sector % 8)
|
|
if [ $modsector -gt 0 ]; then
|
|
vtdebug "modsector:$modsector need to be aligned with 4KB"
|
|
part1_end_sector=$(expr $part1_end_sector - $modsector)
|
|
part2_start_sector=$(expr $part1_end_sector + 1)
|
|
fi
|
|
|
|
part2_end_sector=$(expr $part2_start_sector + $VENTOY_SECTOR_NUM - 1)
|
|
|
|
export part2_start_sector
|
|
|
|
vtdebug "part1_start_sector=$part1_start_sector part1_end_sector=$part1_end_sector"
|
|
vtdebug "part2_start_sector=$part2_start_sector part2_end_sector=$part2_end_sector"
|
|
|
|
if [ -e $PART1 ]; then
|
|
echo "delete $PART1"
|
|
rm -f $PART1
|
|
fi
|
|
|
|
if [ -e $PART2 ]; then
|
|
echo "delete $PART2"
|
|
rm -f $PART2
|
|
fi
|
|
|
|
echo ""
|
|
echo "Create partitions on $DISK by $PARTTOOL in MBR style ..."
|
|
|
|
if [ "$PARTTOOL" = "parted" ]; then
|
|
vtdebug "format disk by parted ..."
|
|
parted -a none --script $DISK \
|
|
mklabel msdos \
|
|
unit s \
|
|
mkpart primary ntfs $part1_start_sector $part1_end_sector \
|
|
mkpart primary fat16 $part2_start_sector $part2_end_sector \
|
|
set 1 boot on \
|
|
quit
|
|
|
|
sync
|
|
echo -en '\xEF' | dd of=$DISK conv=fsync bs=1 count=1 seek=466 > /dev/null 2>&1
|
|
else
|
|
vtdebug "format disk by fdisk ..."
|
|
|
|
fdisk $DISK >>./log.txt 2>&1 <<EOF
|
|
o
|
|
n
|
|
p
|
|
1
|
|
$part1_start_sector
|
|
$part1_end_sector
|
|
n
|
|
p
|
|
2
|
|
$part2_start_sector
|
|
$part2_end_sector
|
|
t
|
|
1
|
|
7
|
|
t
|
|
2
|
|
ef
|
|
a
|
|
1
|
|
w
|
|
EOF
|
|
fi
|
|
|
|
udevadm trigger --name-match=$DISK >/dev/null 2>&1
|
|
partprobe >/dev/null 2>&1
|
|
sleep 3
|
|
echo "Done"
|
|
|
|
|
|
echo 'Wait for partitions ...'
|
|
for i in 0 1 2 3 4 5 6 7 8 9; do
|
|
if [ -b $PART1 -a -b $PART2 ]; then
|
|
break
|
|
else
|
|
echo "Wait for $PART1/$PART2 ..."
|
|
sleep 1
|
|
fi
|
|
done
|
|
|
|
if ! [ -b $PART1 ]; then
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${PART1#/dev/}/dev)
|
|
echo "mknod -m 0660 $PART1 b $MajorMinor ..."
|
|
mknod -m 0660 $PART1 b $MajorMinor
|
|
fi
|
|
|
|
if ! [ -b $PART2 ]; then
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${PART2#/dev/}/dev)
|
|
echo "mknod -m 0660 $PART2 b $MajorMinor ..."
|
|
mknod -m 0660 $PART2 b $MajorMinor
|
|
fi
|
|
|
|
if [ -b $PART1 -a -b $PART2 ]; then
|
|
echo "partition exist OK"
|
|
else
|
|
echo "[FAIL] $PART1/$PART2 does not exist"
|
|
exit 1
|
|
fi
|
|
|
|
echo "create efi fat fs $PART2 ..."
|
|
for i in 0 1 2 3 4 5 6 7 8 9; do
|
|
check_umount_disk "$PART2"
|
|
|
|
if mkfs.vfat -F 16 -n VTOYEFI -s 1 $PART2; then
|
|
echo 'success'
|
|
break
|
|
else
|
|
echo "$? retry ..."
|
|
sleep 2
|
|
fi
|
|
done
|
|
}
|
|
|
|
|
|
format_ventoy_disk_gpt() {
|
|
reserve_mb=$1
|
|
DISK=$2
|
|
PARTTOOL=$3
|
|
|
|
PART1=$(get_disk_part_name $DISK 1)
|
|
PART2=$(get_disk_part_name $DISK 2)
|
|
|
|
sector_num=$(cat /sys/block/${DISK#/dev/}/size)
|
|
|
|
part1_start_sector=2048
|
|
|
|
if [ $reserve_mb -gt 0 ]; then
|
|
reserve_sector_num=$(expr $reserve_mb \* 2048 + 33)
|
|
part1_end_sector=$(expr $sector_num - $reserve_sector_num - $VENTOY_SECTOR_NUM - 1)
|
|
else
|
|
part1_end_sector=$(expr $sector_num - $VENTOY_SECTOR_NUM - 34)
|
|
fi
|
|
|
|
part2_start_sector=$(expr $part1_end_sector + 1)
|
|
|
|
modsector=$(expr $part2_start_sector % 8)
|
|
if [ $modsector -gt 0 ]; then
|
|
vtdebug "modsector:$modsector need to be aligned with 4KB"
|
|
part1_end_sector=$(expr $part1_end_sector - $modsector)
|
|
part2_start_sector=$(expr $part1_end_sector + 1)
|
|
fi
|
|
|
|
part2_end_sector=$(expr $part2_start_sector + $VENTOY_SECTOR_NUM - 1)
|
|
|
|
export part2_start_sector
|
|
|
|
vtdebug "part1_start_sector=$part1_start_sector part1_end_sector=$part1_end_sector"
|
|
vtdebug "part2_start_sector=$part2_start_sector part2_end_sector=$part2_end_sector"
|
|
|
|
if [ -e $PART1 ]; then
|
|
echo "delete $PART1"
|
|
rm -f $PART1
|
|
fi
|
|
|
|
if [ -e $PART2 ]; then
|
|
echo "delete $PART2"
|
|
rm -f $PART2
|
|
fi
|
|
|
|
echo ""
|
|
echo "Create partitions on $DISK by $PARTTOOL in GPT style ..."
|
|
|
|
vtdebug "format disk by parted ..."
|
|
|
|
if [ "$TOOLDIR" != "aarch64" ]; then
|
|
vt_set_efi_type="set 2 msftdata on"
|
|
fi
|
|
|
|
parted -a none --script $DISK \
|
|
mklabel gpt \
|
|
unit s \
|
|
mkpart Ventoy ntfs $part1_start_sector $part1_end_sector \
|
|
mkpart VTOYEFI fat16 $part2_start_sector $part2_end_sector \
|
|
$vt_set_efi_type \
|
|
set 2 hidden on \
|
|
quit
|
|
|
|
sync
|
|
|
|
vtoycli gpt -f $DISK
|
|
sync
|
|
|
|
udevadm trigger --name-match=$DISK >/dev/null 2>&1
|
|
partprobe >/dev/null 2>&1
|
|
sleep 3
|
|
echo "Done"
|
|
|
|
echo 'Wait for partitions ...'
|
|
for i in 0 1 2 3 4 5 6 7 8 9; do
|
|
if [ -b $PART1 -a -b $PART2 ]; then
|
|
break
|
|
else
|
|
echo "Wait for $PART1/$PART2 ..."
|
|
sleep 1
|
|
fi
|
|
done
|
|
|
|
if ! [ -b $PART1 ]; then
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${PART1#/dev/}/dev)
|
|
echo "mknod -m 0660 $PART1 b $MajorMinor ..."
|
|
mknod -m 0660 $PART1 b $MajorMinor
|
|
fi
|
|
|
|
if ! [ -b $PART2 ]; then
|
|
MajorMinor=$(sed "s/:/ /" /sys/class/block/${PART2#/dev/}/dev)
|
|
echo "mknod -m 0660 $PART2 b $MajorMinor ..."
|
|
mknod -m 0660 $PART2 b $MajorMinor
|
|
fi
|
|
|
|
if [ -b $PART1 -a -b $PART2 ]; then
|
|
echo "partition exist OK"
|
|
else
|
|
echo "[FAIL] $PART1/$PART2 does not exist"
|
|
exit 1
|
|
fi
|
|
|
|
echo "create efi fat fs $PART2 ..."
|
|
|
|
for i in 0 1 2 3 4 5 6 7 8 9; do
|
|
check_umount_disk "$PART2"
|
|
|
|
if mkfs.vfat -F 16 -n VTOYEFI -s 1 $PART2; then
|
|
echo 'success'
|
|
break
|
|
else
|
|
echo "$? retry ..."
|
|
sleep 2
|
|
fi
|
|
done
|
|
}
|
|
|
|
|
|
|
|
|
|
|