setup-disk.in 22 KB
Newer Older
Natanael Copa's avatar
Natanael Copa committed
1 2 3 4 5
#!/bin/sh

PREFIX=
. "$PREFIX/lib/libalpine.sh"

6
MBR=${MBR:-"/usr/share/syslinux/mbr.bin"}
7 8
ROOTFS=${ROOTFS:-ext4}
BOOTFS=${BOOTFS:-ext4}
9
VARFS=${VARFS:-ext4}
Natanael Copa's avatar
Natanael Copa committed
10 11 12 13 14 15 16 17 18 19 20

in_list() {
	local i="$1"
	shift
	while [ $# -gt 0 ]; do
		[ "$i" = "$1" ] && return 0
		shift
	done
	return 1
}

21 22 23 24 25 26 27 28 29 30 31
all_in_list() {
	local needle="$1"
	local i
	[ -z "$needle" ] && return 1
	shift
	for i in $needle; do
		in_list "$i" $@ || return 1
	done
	return 0
}

32 33 34 35 36 37 38 39
# wrapper to only show given device
_blkid() {
	blkid | grep "^$1:"
}

# if given device have an UUID display it, otherwise return the device
uuid_or_device() {
	local i=
40 41 42
	case "$1" in
		/dev/md*) echo "$1" && return 0;;
	esac
43 44 45 46 47 48 49 50 51 52 53 54 55
	for i in $(_blkid "$1"); do
		case "$i" in
			UUID=*) eval $i;;
		esac
	done
	if [ -n "$UUID" ]; then
		echo "UUID=$UUID"
	else
		echo "$1"
	fi
}

# generate an fstab from a given mountpoint. Convert to UUID if possible
56
enumerate_fstab() {
57
	local mnt="$1"
58
	local fs_spec= fs_file= fs_vfstype= fs_mntops= fs_freq= fs_passno=
59
	[ -z "$mnt" ] && return
60 61
	local escaped_mnt=$(echo $mnt | sed -e 's:/*$::' -e 's:/:\\/:g')
	awk "\$2 ~ /^$escaped_mnt(\/|\$)/ {print \$0}" /proc/mounts | \
62 63 64 65
		sed "s:$mnt:/:g; s: :\t:g" | sed 's:/\+:/:g' | \
		while read fs_spec fs_file fs_vfstype fs_mntops fs_freq fs_passno; do
			echo -e "$(uuid_or_device $fs_spec)\t${fs_file}\t${fs_vfstype}\t${fs_mntops} ${fs_freq} ${fs_passno}"
		done
66 67
}

68 69 70 71 72
is_vmware() {
	grep -q VMware /proc/scsi/scsi 2>/dev/null \
		|| grep -q VMware /proc/ide/hd*/model 2>/dev/null
}

73 74 75 76
is_xen() {
	[ -d /proc/xen ]
}

77 78 79 80 81
# return true (0) if given device is lvm
is_lvm() {
	lvs "$1" >/dev/null 2>&1
}

82 83
# Find the disk device from given partition
disk_from_part() {
84 85
	# we need convert cciss/c0d0* cciss!c0d0*...
	local i= part=$(echo ${1#/dev/} | sed 's:/:!:g')
86 87
	for i in /sys/block/*/$part; do
		i=${i%/*}
88
		# ...and back from cciss!c0d0 to cciss/c0d0
89 90 91 92
		if [ -b "/dev/${i##*/}" ]; then
			echo "/dev/${i##*/}" | sed 's:!:/:g'
			return 0
		fi
93 94 95 96
	done
	return 1
}

97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131
unpack_apkovl() {
	local ovl="$1"
	local dest="$2"
	local suffix=${ovl##*.}
	local i
	ovlfiles=/tmp/ovlfiles
	if [ "$suffix" = "gz" ]; then
		if ! tar -C "$dest" --numeric-owner -zxvf "$ovl" > $ovlfiles; then
			echo -n "Continue anyway? [Y/n]: "
			read i
			case "$i" in
				n*|N*) return 1;;
			esac
		fi
		return 0 
	fi

	apk add -q openssl

	if ! openssl list-cipher-commands | grep "^$suffix$" > /dev/null; then
		errstr="Cipher $suffix is not supported"
		return 1
	fi
	local count=0
	# beep
	echo -e "\007"
	while [ $count -lt 3 ]; do
		openssl enc -d -$suffix -in "$ovl" | tar --numeric-owner \
			-C "$dest" -zxv >$ovlfiles 2>/dev/null && return 0
		count=$(( $count + 1 ))
	done
	ovlfiles=
	return 1
}

132 133 134 135 136 137 138 139
# find filesystem of given mounted dir
find_mount_fs() {
	local mount_point="$1"
	awk "\$2 == \"$mount_point\" {print \$3}" /proc/mounts | tail -n 1
}

# find device for given mounted dir
find_mount_dev() {
140
	local mnt="$1"
141 142 143 144
	awk "\$2 == \"$mnt\" { print \$1 }" /proc/mounts | tail -n 1
}

supported_boot_fs() {
145
	local supported="ext2 ext3 ext4 btrfs"
146 147 148 149 150 151 152 153
	local fs=
	for fs in $supported; do
		[ "$fs" = "$1" ] && return 0
	done
	echo "$1 is not supported. Only supported are: $supported" >&2
	return 1
}

154 155 156 157 158 159 160 161 162 163
find_volume_group() {
	local lv=${1##*/}
	lvs --noheadings "$1" | awk "\$1 == \"$lv\" {print \$2}"
}

find_pvs_in_vg() {
	local vg="$1"
	pvs --noheadings | awk "\$2 == \"$vg\" {print \$1}"
}

164 165
install_mounted_root() {
	local mnt="$1" mnt_boot="$1" boot_fs= root_fs=
166
	local initfs_features="ata base ide scsi usb virtio"
167 168
	local pvs= dev= rootdev= bootdev= raidopt= root= modules=
	local kernel_opts="quiet"
169

170
	rootdev=$(find_mount_dev "$mnt")
171 172 173 174
	if [ -z "$rootdev" ]; then
		echo "$mnt does not seem to be a mount point" >&2
		return 1
	fi
175
	root_fs=$(find_mount_fs "$mnt")
176 177 178 179
	initfs_features="$initfs_features $root_fs"

	if is_lvm "$rootdev"; then
		initfs_features="$initfs_features lvm"
180 181
		local vg=$(find_volume_group "$rootdev")
		pvs=$(find_pvs_in_vg $vg)
182 183
	fi

184

185 186 187 188 189 190 191 192 193 194
	bootdev=$(find_mount_dev "$mnt"/boot)
	if [ -z "$bootdev" ]; then
		bootdev=$rootdev
	else
		mnt_boot="$mnt"/boot
		bootdev=$(find_mount_dev "$mnt_boot")
	fi
	boot_fs=$(find_mount_fs "$mnt_boot")
	supported_boot_fs "$boot_fs" || return 1

195 196 197
	if [ -e "/sys/block/${bootdev#/dev/}/md" ]; then
		raidopt="--raid"
	fi
198 199 200 201 202 203 204 205 206 207 208 209

	for dev in $rootdev $pvs; do
		[ -e "/sys/block/${dev#/dev/}/md" ] || continue

		local md=${dev#/dev/}
		initfs_features="$(echo $initfs_features | sed 's/raid//') raid"
		local level=$(cat /sys/block/$md/md/level)
		case "$level" in
			raid1) raidmod="$raidmod,$level";;
			raid[456]) raidmod="$raidmod,raid456";;
		esac
	done
210 211 212 213 214 215 216 217

	# check if we need hardware raid drivers
	case $rootdev in
		/dev/cciss/*)
			initfs_features="$(echo $initfs_features | sed 's/raid//') raid"
			;;
	esac

218 219 220 221 222 223
	if [ -n "$VERBOSE" ]; then
		echo "Root device:     $rootdev"
		echo "Root filesystem: $root_fs"
		echo "Boot device:     $bootdev"
		echo "Boot filesystem: $boot_fs"
	fi
224

225
	if [ -z "$APKOVL" ]; then
226 227
		ovlfiles=/tmp/ovlfiles
		lbu package - | tar -C "$mnt" -zxv > "$ovlfiles"
228 229
	else
		echo "Restoring backup from $APKOVL to $rootdev..."
230
		unpack_apkovl "$APKOVL" "$mnt" || return 1
231
	fi
232

233
	# generate mkinitfs.conf
234
	mkdir -p "$mnt"/etc/mkinitfs/files.d
235
	echo "features=\"$initfs_features\"" > "$mnt"/etc/mkinitfs/mkinitfs.conf
236 237 238 239
	if [ -n "$raidmod" ]; then
		echo "/sbin/mdadm" > "$mnt"/etc/mkinitfs/files.d/raid
		echo "/etc/mdadm.conf" >> "$mnt"/etc/mkinitfs/files.d/raid
	fi
240

241 242 243 244 245 246 247 248 249 250 251 252 253 254 255
	# generate update-extlinux.conf
	root=$(uuid_or_device $rootdev)
	if is_vmware; then
		kernel_opts="pax_nouderef $kernel_opts"
	fi
	if is_xen; then
		kernel_opts="xen $kernel_opts"
	fi
	modules="sd-mod,usb-storage,${root_fs}${raidmod}"
	sed -e "s:^root=.*:root=$root:" \
		-e "s:^default_kernel_opts=.*:default_kernel_opts=\"$kernel_opts\":" \
		-e "s:^modules=.*:modules=$modules:" \
		/etc/update-extlinux.conf > "$mnt"/etc/update-extlinux.conf


256 257 258 259 260 261 262 263 264 265 266
	# generate the fstab
	if [ -f "$mnt"/etc/fstab ]; then
		mv "$mnt"/etc/fstab "$mnt"/etc/fstab.old
	fi
	enumerate_fstab "$mnt" >> "$mnt"/etc/fstab
	cat >>"$mnt"/etc/fstab <<EOF
/dev/cdrom	/media/cdrom	iso9660	noauto,ro 0 0
/dev/fd0	/media/floppy	vfat	noauto	0 0
/dev/usbdisk	/media/usb	vfat	noauto	0 0
EOF

267
	# remove the installed db in case its there so we force re-install
268
	rm -f "$mnt"/var/lib/apk/installed "$mnt"/lib/apk/db/installed
269
	echo "Installing system on $rootdev:"
270 271 272 273
	# apk reads config from target root so we need to copy the config
	mkdir -p "$mnt"/etc/apk/keys/
	cp /etc/apk/keys/* "$mnt"/etc/apk/keys/

274

275
	local apkflags="--initdb --quiet --progress --update-cache --clean-protected"
276
	local pkgs=$(cat "$mnt"/etc/apk/world "$mnt"/var/lib/apk/world 2>/dev/null)
277
	pkgs="$pkgs acct linux-$KERNEL_FLAVOR alpine-base"
278 279 280 281 282
	local repos=$(sed -e 's/\#.*//' /etc/apk/repositories)
	local repoflags=
	for i in $repos; do
		repoflags="$repoflags --repository $i"
	done
283 284
	
	apk add --root "$mnt" $apkflags --overlay-from-stdin \
285
		$repoflags $pkgs <$ovlfiles>/dev/null || return 1
286
	echo ""
287
	is_xen || extlinux $raidopt --install "$mnt"/boot
288 289 290 291
}

unmount_partitions() {
	local mnt="$1"
292 293 294

	# unmount the partitions
	umount $(awk '{print $2}' /proc/mounts | grep ^"$mnt" | sort -r)
295
}
296

297 298 299
# figure out decent default swap size in mega bytes
find_swap_size() {
	local memtotal_kb=$(awk '$1 == "MemTotal:" {print $2}' /proc/meminfo)
300 301 302 303
	# use 2 * avaiable ram or no more than 1/3 of smallest disk space
	local size=$(( $memtotal_kb * 2 / 1024 ))
	local disk= disksize=
	for disk in $@; do
304 305
		local sysfsdev=$(echo ${disk#/dev/} | sed 's:/:!:g')
		local sysfspath=/sys/block/$sysfsdev/size
306 307 308 309 310 311 312 313 314 315 316 317 318 319 320
		# disksize = x * 512 / (1024 * 1024) = x / 2048
		# maxsize = $disksize / 4 = x / (2048 * 4) = x / 8192
		maxsize=$(awk '{ printf "%i", $0 / 8192 }' $sysfspath )
		if [ $size -gt $maxsize ]; then
			size=$maxsize
		fi
	done
	if [ $size -gt 4096 ]; then
		# dont ever use more than 4G
		size=4096
	elif [ $size -lt 64 ]; then
		# dont bother create swap smaller than 64MB
		size=0
	fi
	echo $size
321 322 323 324
}

has_mounted_part() {
	local p
325
	local sysfsdev=$(echo ${1#/dev/} | sed 's:/:!:g')
Natanael Copa's avatar
Natanael Copa committed
326
	# parse /proc/mounts for mounted devices
327
	for p in $(awk '$1 ~ /^\/dev\// {gsub("/dev/", "", $1); gsub("/", "!", $1); print $1}' \
328
			/proc/mounts); do
329 330
		[ "$p" = "$sysfsdev" ] && return 0
		[ -e /sys/block/$sysfsdev/$p ] && return 0
331 332 333 334
	done
	return 1
}

Natanael Copa's avatar
Natanael Copa committed
335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354
has_holders() {
	local i
	# check if device is used by any md devices
	for i in $1/holders/* $1/*/holders/*; do
		[ -e "$i" ] && return 0
	done
	return 1
}

is_available_disk() {
	local dev=$1
	local b=$(echo $p | sed 's:/:!:g')

	# check if its a "root" block device and not a partition
	[ -e /sys/block/$b ] || return 1
	
	# check so it does not have mounted partitions
	has_mounted_part $dev && return 1

	# check so its not part of an md setup
355 356 357 358
	if has_holders /sys/block/$b; then
		[ -n "$USE_RAID" ] && echo "Warning: $dev is part of a running raid" >&2
		return 1
	fi
Natanael Copa's avatar
Natanael Copa committed
359 360 361 362 363 364 365

	# check so its not an md device
	[ -e /sys/block/$b/md ] && return 1

	return 0
}

366
find_disks() {
Natanael Copa's avatar
Natanael Copa committed
367
	local p=
368
	for p in $(awk '$1 ~ /[0-9]+/ {print $4}' /proc/partitions); do
Natanael Copa's avatar
Natanael Copa committed
369
		is_available_disk $p && echo -n " $p"
370 371 372
	done
}

373 374 375 376 377 378 379 380 381 382 383
stop_all_raid() {
	local rd
	for rd in /dev/md*; do
		[ -b $rd ] && mdadm --stop $rd
	done
}

# install needed programs
init_progs() {
	local raidpkg=
	[ -n "$USE_RAID" ] && raidpkg="mdadm"
384
	apk add -q sfdisk e2fsprogs lvm2 $raidpkg syslinux $@
385 386
}

387
show_disk_info() {
Natanael Copa's avatar
Natanael Copa committed
388
	local disk= vendor= model= d= size=
389
	for disk in $@; do
390 391
		local dev=${disk#/dev/}
		d=$(echo $dev | sed 's:/:!:g')
392 393 394
		vendor=$(cat /sys/block/$d/device/vendor 2>/dev/null)
		model=$(cat /sys/block/$d/device/model 2>/dev/null)
		size=$(awk '{gb = ($1 * 512)/1000000000; printf "%.1f GB\n", gb}' /sys/block/$d/size 2>/dev/null)
395
		echo "  $dev	($size $vendor $model)"
396 397
	done
}
398

399 400
confirm_erase() {
	local answer=
401
	echo "WARNING: The following disk(s) will be erased:"
402
	show_disk_info $@
403
	echo -n "WARNING: Erase the above disk(s) and continue? [y/N]: "
404
	
405 406
	read answer
	case "$answer" in
407
		y*|Y*) return 0;;
408
	esac
409
	return 1
410 411 412 413 414 415 416 417 418
}

# setup disk dev in $1 for LVM usage.
# usage: setup_partitions <diskdev> size1,type1 [size2,type2 ...] 
setup_partitions() {
	local diskdev="$1"
	shift

	echo "Initializing partitions on $diskdev..."
419 420 421 422 423 424 425

	# new disks does not have an DOS signature in sector 0
	# this makes sfdisk complain. We can workaround this by letting
	# fdisk create that DOS signature, by just do a "w", a write.
	# http://bugs.alpinelinux.org/issues/show/145
	echo "w" | fdisk $diskdev >/dev/null

426 427 428
	# fix the MBR while here so extlinux can boot
	cat "$MBR" > $diskdev

429 430
	local start=0
	local line=
431 432
	# create new partitions
	(
433 434 435 436
		for line in "$@"; do
			echo "$start,$line"
			start=
		done
437 438 439 440 441 442 443 444 445 446 447 448
	) | sfdisk -q -L -uM $diskdev >>/tmp/sfdisk.out || return 1

	# create device nodes if not exist
	mdev -s
}

# find the bootable partition on given disk
find_boot_partition() {
	sfdisk -d $1 | awk '/bootable/ {print $1}'
}

# find the partition(s) for LVM
449
# this is not marked as bootable and is type 8e
450 451 452 453 454
find_lvm_partition() {
	local type=8e
	sfdisk -d $1 | grep -v bootable | awk "/Id=$type/ {print \$1}"
}

455
# set up boot device. We only use raid1 for boot devices if any raid
456
setup_boot_dev() {
457 458 459 460
	local disk= bootdev=
	local part=$(for disk in $@; do find_boot_partition $disk; done)
	set -- $part
	bootdev=$1
461
	[ -z "$bootdev" ] && return 1
462
	echo "Creating file systems..."
463
	if [ -n "$USE_RAID" ]; then
464 465 466 467 468 469 470 471
		local missing=
		local num=$#
		if [ $# -eq 1 ]; then
			missing="missing"
			num=2
		fi
		mdadm --create /dev/md0 --level=1 --raid-devices=$num \
			--metadata=0.90 --quiet --run $@ $missing || return 1
472 473
		bootdev=/dev/md0
	fi
474
	mkfs.$BOOTFS -q $bootdev
475 476 477 478
	BOOT_DEV="$bootdev"
}

# $1 = index
479 480 481
# $2 = partition type
# $3... = disk devices
find_nth_non_boot_parts() {
482
	local idx=$1
483
	local id=$2
484 485
	local disk=
	shift
486
	shift
487 488
	for disk in $@; do
		sfdisk -d $disk | grep -v bootable \
489
			| awk "/Id=$id/ { i++; if (i==$idx) print \$1 }"
490 491 492 493 494 495 496 497 498 499
	done
}

setup_non_boot_raid_dev() {
	local md_dev=$1
	local idx=${md_dev#/dev/md}
	shift
	local level=1
	local numdevs=$#
	local missing=
500
	local raid_parts=$(find_nth_non_boot_parts $idx "fd" $@)
501 502 503 504 505 506 507 508 509 510
	set -- $raid_parts
	# how many disks do we have?
	case $# in
		0) echo "No Raid partitions found" >&2; return 1;;
		1) level=1; missing="missing"; num=2;;
		2) level=1; missing=  ; num=2;;
		*) level=5; missing=  ; num=$#;;
	esac
	mdadm --create /dev/md$idx --level=$level --raid-devices=$num \
		--quiet --run $@ $missing || return 1
511 512 513 514
}

# setup device for lvm, create raid array if needed
setup_lvm_volume_group() {
515 516 517
	local vgname="$1"
	shift
	local lvmdev=
518 519

	if [ -n "$USE_RAID" ]; then
520 521 522 523
		setup_non_boot_raid_dev /dev/md1 $@ || return 1
		lvmdev=/dev/md1
	else	
		lvmdev=$(find_lvm_partition $1)
524 525 526 527 528
	fi

	# be quiet on success
	local errmsg=$(dd if=/dev/zero of=$lvmdev bs=1k count=1 2>&1) \
		|| echo "$errmsg"
529 530 531 532 533 534 535 536 537 538 539 540 541 542
	pvcreate --quiet $lvmdev \
		&& vgcreate --quiet $vgname $lvmdev >/dev/null
}

# set up swap on given device(s)
setup_swap_dev() {
	local swap_dev=
	sed -i -e '/swap/d' /etc/fstab
	for swap_dev in "$@"; do
		mkswap $swap_dev >/dev/null
		echo -e "$swap_dev\tswap\t\tswap\tdefaults 0 0" >> /etc/fstab
	done
	swapon -a
	rc-update --quiet add swap boot
543 544 545
}

# setup and enable swap on given volumegroup if needed
546
setup_lvm_swap() {
547
	local vgname="$1"
548
	local swapname=lv_swap
549 550 551
	if [ -z "$SWAP_SIZE" ] || [ "$SWAP_SIZE" -eq 0 ]; then
		return
	fi
552 553
	lvcreate --quiet -n $swapname -L ${SWAP_SIZE}MB $vgname
	setup_swap_dev /dev/$vgname/$swapname
554 555 556 557 558 559 560 561 562 563
}

# if /var is mounted, move out data and umount it
reset_var() {
	[ -z "$(find_mount_dev /var)" ] && return 0
	mkdir /.var
	mv /var/* /.var/ 2>/dev/null
	umount /var && 	rm -rf /var && mv /.var /var && rm -rf /var/lost+found
}

564 565 566
# set up /var on given device
setup_var() {
	local var_dev="$1"
567
	local varfs=${VARFS}
568
	echo "Creating file systems..."
569 570 571 572 573 574 575 576 577 578 579
	mkfs.$varfs -q $var_dev >/dev/null || return 1
	sed -i -e '/[[:space:]]\/var[[:space:]]/d' /etc/fstab
	echo -e "${var_dev}\t/var\t\t${varfs}\tdefaults 1 2" >> /etc/fstab

	mv /var /.var
	mkdir /var
	mount /var
	mv /.var/* /var/
	rmdir /.var

	/etc/init.d/syslog --quiet restart
580
	setup_mdadm_conf
581 582
}

583
setup_mdadm_conf() {
584
	local mods= mod=
585 586 587
	if [ -n "$USE_RAID" ]; then
		mdadm --detail --scan > /etc/mdadm.conf
		rc-update --quiet add mdadm-raid boot
588 589 590 591 592 593
		mods=$(awk '/^raid/ {print $1}' /proc/modules)
		for mod in $mods; do
			if ! grep -q "^$mod" /etc/modules; then
				echo $mod >> /etc/modules
			fi
		done
594 595 596
	fi
}

597
data_only_disk_install_lvm() {
598
	local diskdev=
599 600
	local vgname=vg0
	local var_dev=/dev/$vgname/lv_var
601 602 603 604
	local lvm_part_type="8e"
	local raid_part_type="fd"
	local part_type=$lvm_part_type
	local size=
605 606

	init_progs || return 1
607
	confirm_erase $@ || return 1
608

609 610 611 612
	if [ "$USE_RAID" ]; then
		part_type=$raid_part_type
		stop_all_raid
	fi
613

614 615 616 617 618 619
	for diskdev in "$@"; do
		setup_partitions $diskdev "$size,$part_type" || return 1
	done
	
	setup_lvm_volume_group $vgname $@ || return 1
	setup_lvm_swap $vgname
620
	lvcreate --quiet -n ${var_dev##*/} -l 100%FREE $vgname
621
	setup_mdadm_conf
Natanael Copa's avatar
Natanael Copa committed
622
	rc-update add lvm boot
623
	setup_var $var_dev
624 625
}

626 627
data_only_disk_install() {
	local diskdev=
628 629
	local var_dev=
	local var_part_type="83"
630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661
	local swap_part_type=82
	local size=
	local swap_dev= var_dev=

	init_progs || return 1
	confirm_erase $@ || return 1

	if [ "$USE_RAID" ]; then
		var_part_type="fd"
		swap_part_type="fd"
		stop_all_raid
	fi

	for diskdev in "$@"; do
		setup_partitions $diskdev \
			"$SWAP_SIZE,$swap_part_type" \
			"$size,$var_part_type" || return 1
	done

	if [ "$USE_RAID" ]; then
		[ $SWAP_SIZE -gt 0 ] && setup_non_boot_raid_dev /dev/md1 $@
		setup_non_boot_raid_dev /dev/md2 $@ || return 1
		swap_dev=/dev/md1
		var_dev=/dev/md2
	else
		swap_dev=$(find_nth_non_boot_parts 1 82 $@)
		var_dev=$(find_nth_non_boot_parts 1 83 $@)
	fi
	[ $SWAP_SIZE -gt 0 ] && setup_swap_dev $swap_dev
	setup_var $var_dev
}

662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678
# setup 
setup_root() {
	local root_dev="$1" boot_dev="$2"
	mkfs.$ROOTFS -q "$root_dev"
	mkdir -p /mnt
	mount -t $ROOTFS $root_dev /mnt || return 1
	if [ -n "$boot_dev" ]; then
		mkdir -p /mnt/boot
		mount -t $BOOTFS $boot_dev /mnt/boot || return 1
	fi

	setup_mdadm_conf
	install_mounted_root /mnt || return 1
	unmount_partitions /mnt
	swapoff -a

	echo ""
679
	echo "Installation is complete. Please reboot."
680 681
}

682
native_disk_install_lvm() {
683 684 685 686 687 688 689 690 691 692
	local diskdev= vgname=vg0
	local lvm_part_type="8e"
	local raid_part_type="fd"
	local boot_part_type="83"
	local boot_size=${BOOT_SIZE:-100}
	local lvm_size=
	local root_dev=/dev/$vgname/lv_root

	init_progs syslinux || return 1
	confirm_erase $@ || return 1
693 694

	if [ -n "$USE_RAID" ]; then
695 696 697
		boot_part_type="fd"
		lvm_part_type="fd"
		stop_all_raid
698
	fi
699 700 701 702 703 704 705 706 707 708 709 710
	for diskdev in "$@"; do
		setup_partitions $diskdev \
			"$boot_size,$boot_part_type,*" \
			"$lvm_size,$lvm_part_type" || return 1
	done

	# will find BOOT_DEV for us
	setup_boot_dev $@

	setup_lvm_volume_group $vgname $@ || return 1
	setup_lvm_swap $vgname
	lvcreate --quiet -n ${root_dev##*/} -l 100%FREE $vgname
Natanael Copa's avatar
Natanael Copa committed
711
	rc-update add lvm boot
712
	setup_root $root_dev $BOOT_DEV
Natanael Copa's avatar
Natanael Copa committed
713 714
}

715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754
native_disk_install() {
	local root_part_type="83" swap_part_type="82" boot_part_type="83"
	local boot_size=${BOOT_SIZE:-100}
	local swap_size=${SWAP_SIZE}
	local root_size=
	local root_dev= boot_dev= swap_dev=

	init_progs syslinux || return 1
	confirm_erase $@ || return 1

	if [ -n "$USE_RAID" ]; then
		boot_part_type="fd"
		root_part_type="fd"
		swap_part_type="fd"
		stop_all_raid
	fi
	for diskdev in "$@"; do
		setup_partitions $diskdev \
			"$boot_size,$boot_part_type,*" \
			"$swap_size,$swap_part_type" \
			"$root_size,$root_part_type" \
			|| return 1
	done

	# will find BOOT_DEV for us
	setup_boot_dev $@
	
	if [ "$USE_RAID" ]; then
		[ $SWAP_SIZE -gt 0 ] && setup_non_boot_raid_dev /dev/md1 $@
		setup_non_boot_raid_dev /dev/md2 $@ || return 1
		swap_dev=/dev/md1
		root_dev=/dev/md2
	else
		swap_dev=$(find_nth_non_boot_parts 1 82 $@)
		root_dev=$(find_nth_non_boot_parts 1 83 $@)
	fi
	[ $SWAP_SIZE -gt 0 ] && setup_swap_dev $swap_dev
	setup_root $root_dev $BOOT_DEV
}

755
diskselect_help() {
756 757
	cat <<__EOF__

Natanael Copa's avatar
Natanael Copa committed
758
The disk you select can be used for a traditional disk install or for a
759 760 761 762
data-only install.

The disk will be erased.

Natanael Copa's avatar
Natanael Copa committed
763
Enter 'none' if you want to run diskless.
764 765 766 767

__EOF__
}

768
diskmode_help() {
769 770
	cat <<__EOF__

771
You can select between 'sys' or 'data'.
772

Natanael Copa's avatar
Natanael Copa committed
773 774 775 776 777
sys:
  This mode is a traditional disk install. The following partitions will be
  created on the disk: /boot, / (filesystem root) and swap.
    
  This mode may be used for development boxes, desktops, virtual servers, etc.
778

Natanael Copa's avatar
Natanael Copa committed
779 780 781
data:
  This mode uses your disk(s) for data storage, not for the operating system.
  The system itself will run from tmpfs (RAM).
782

Natanael Copa's avatar
Natanael Copa committed
783 784
  Use this mode if you only want to use the disk(s) for a mailspool, databases,
  logs, etc.
785 786 787 788 789 790 791 792 793

__EOF__
}

# ask for a root or data disk
# returns answer in global variable $answer
ask_disk() {
	local prompt="$1"
	local help_func="$2"
794
	local i=
795 796
	shift 2
	answer=
797
	local default_disk=${DEFAULT_DISK:-$1}
798

799 800 801
	while ! all_in_list "$answer" $@ "none" "abort"; do
		echo "Available disks are:"
		show_disk_info "$@"
802 803
		echon "$prompt [$default_disk] "
		default_read answer $default_disk
804 805 806 807
		case "$answer" in
			'abort') exit 0;;
			'none') return 0;;
			'?') $help_func;;
808 809 810 811 812 813
			*) for i in $answer; do
				if ! [ -b "/dev/$i" ]; then
					echo "/dev/$i is not a block device" >&2
					answer=
				   fi
			done;;
814 815 816 817
		esac
	done
}

818 819 820 821 822 823 824 825 826 827 828 829 830 831 832 833 834 835 836
usage() {
	cat <<__EOF__
usage: setup-disk [-hqr] [-k kernelflavor] [-m MODE] [-o apkovl] [-s SWAPSIZE]
		  [MOUNTPOINT | DISKDEV...]

Install alpine on harddisk.

If MOUNTPOINT is specified, then do a traditional disk install with MOUNTPOINT
as root.

If DISKDEV is specified, then use the specified disk(s) without asking. If
multiple disks are specified then set them up in a RAID array. If there are
mode than 2 disks, then use raid level 5 instead of raid level 1.

options:
 -h  Show this help
 -m  Use disk for MODE without asking, where MODE is either 'data' or 'root'
 -o  Restore system from given apkovl file
 -k  Use kernelflavor instead of $KERNEL_FLAVOR
837
 -L  Use LVM to manage partitions
838 839 840 841 842 843 844 845
 -q  Exit quietly if no disks are found
 -r  Enable software raid1 with single disk
 -s  Use SWAPSIZE MB instead of $SWAP_SIZE MB for swap (Use 0 to disable swap)
 -v  Be more verbose about what is happening
__EOF__
	exit 1
}

846 847 848
KERNEL_FLAVOR=grsec
case "$(uname -r)" in
	*-vs[0-9]*) KERNEL_FLAVOR=vserver;;
849
	*-pae) KERNEL_FLAVOR=pae;;
850 851
esac

852
DISK_MODE=
853
USE_LVM=
854
# Parse args
855
while getopts "hk:Lm:o:qrs:v" opt; do
856
	case $opt in
857
		m) DISK_MODE="$OPTARG";;
858
		k) KERNEL_FLAVOR="$OPTARG";;
859
		L) USE_LVM="_lvm";;
860
		o) APKOVL="$OPTARG";;
861 862
		q) QUIET=1;;
		r) USE_RAID=1;;
863
		s) SWAP_SIZE="$OPTARG";;
864
		v) VERBOSE=1;;
Natanael Copa's avatar
Natanael Copa committed
865
		*) usage;;
866 867
	esac
done
868
shift $(( $OPTIND - 1))
869

870 871
if [ -d "$1" ]; then
	# install to given mounted root
872
	apk add -q syslinux
873 874
	install_mounted_root "${1%/}" \
		&& echo "You might need fix the MBR to be able to boot" >&2
875 876 877
	exit $?
fi

878 879 880 881 882 883 884 885 886 887
reset_var
swapoff -a

# stop all volume groups in use
vgchange --ignorelockingfailure -a n >/dev/null 2>&1

if [ -n "$USE_RAID" ]; then
	stop_all_raid
fi

888
disks=$(find_disks)
889
diskdevs=
Natanael Copa's avatar
Natanael Copa committed
890 891

# no disks so lets exit quietly.
892 893 894 895
if [ -z "$disks" ]; then
	[ -z "$QUIET" ] && echo "No disks found." >&2
	exit 0
fi
Natanael Copa's avatar
Natanael Copa committed
896

897 898 899 900 901
if [ $# -gt 0 ]; then
	# check that they are 
	for i in "$@"; do
		j=$(readlink -f "$i" | sed 's:^/dev/::; s:/:!:g')
		if ! [ -e "/sys/block/$j/device" ]; then
Natanael Copa's avatar
Natanael Copa committed
902
			echo "$i is not a suitable for partitioning"
903 904
			exit 1
		fi
905
		diskdevs="$diskdevs /dev/$j"
906 907
	done
else
908
	ask_disk "Which disk(s) would you like to use? (or '?' for help or 'none')" \
909
		diskselect_help $disks
910
	if [ "$answer" != none ]; then
911 912 913
		for i in $answer; do
			diskdevs="$diskdevs /dev/$i"
		done
914 915
	else
		DISK_MODE="none"
916
	fi
917
fi
Natanael Copa's avatar
Natanael Copa committed
918

919
if [ -n "$diskdevs" ] && [ -z "$DISK_MODE" ]; then
920
	answer=
921
	disk_is_or_disks_are="disk is"
922
	it_them="it"
923
	set -- $diskdevs
924
	if [ $# -gt 1 ]; then
925
		disk_is_or_disks_are="disks are"
926 927 928
		it_them="them"
	fi
		
929
	while true; do
930
		echo "The following $disk_is_or_disks_are selected:"
931
		show_disk_info $diskdevs
932
		echon "How would you like to use $it_them? ('sys', 'data' or '?' for help) [?] "
933 934 935
		default_read answer '?'
		case "$answer" in
		'?') diskmode_help;;
936
		sys|data) break;;
937 938
		esac
	done
939
	DISK_MODE="$answer"
940
fi
Natanael Copa's avatar
Natanael Copa committed
941

942 943 944 945
if [ -z "$SWAP_SIZE" ]; then
	SWAP_SIZE=$(find_swap_size $diskdevs)
fi

946 947 948 949
set -- $diskdevs
if [ $# -gt 1 ]; then
	USE_RAID=1
fi
950

951 952
dmesg -n1

953
# native disk install
954
case "$DISK_MODE" in
955
sys) native_disk_install$USE_LVM $diskdevs;;
956
data) data_only_disk_install$USE_LVM $diskdevs;;
957
none) exit 0;;
958
*) echo "Not a valid install mode: $DISK_MODE" >&2; exit 1;;
959
esac
Natanael Copa's avatar
Natanael Copa committed
960

961
echo "$DISK_MODE" > /tmp/alpine-install-diskmode.out