setup-disk.in 22.4 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
# default location for mounted root
SYSROOT=${SYSROOT:-/mnt}

Natanael Copa's avatar
Natanael Copa committed
14 15 16 17 18 19 20 21 22 23
in_list() {
	local i="$1"
	shift
	while [ $# -gt 0 ]; do
		[ "$i" = "$1" ] && return 0
		shift
	done
	return 1
}

24 25 26 27 28 29 30 31 32 33 34
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
}

35 36 37 38 39 40 41 42
# 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=
43 44 45
	case "$1" in
		/dev/md*) echo "$1" && return 0;;
	esac
46 47 48 49 50 51 52 53 54 55 56 57 58
	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
59
enumerate_fstab() {
60
	local mnt="$1"
61
	local fs_spec= fs_file= fs_vfstype= fs_mntops= fs_freq= fs_passno=
62
	[ -z "$mnt" ] && return
63 64
	local escaped_mnt=$(echo $mnt | sed -e 's:/*$::' -e 's:/:\\/:g')
	awk "\$2 ~ /^$escaped_mnt(\/|\$)/ {print \$0}" /proc/mounts | \
65 66 67 68
		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
69 70
}

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

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

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

96 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
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
}

131 132 133 134 135 136 137 138
# 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() {
139
	local mnt="$1"
140 141 142 143
	awk "\$2 == \"$mnt\" { print \$1 }" /proc/mounts | tail -n 1
}

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

153 154 155 156 157 158 159 160 161 162
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}"
}

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

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

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

183

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

194 195
	# Check if we boot from raid so we can pass proper option to
	# extlinux later.
196
	if [ -e "/sys/block/${bootdev#/dev/}/md" ]; then
197
		extlinux_raidopt="--raid"
198
	fi
199

200 201
	# check if our root is on raid so we can feed mkinitfs and
	# update-exlinux.conf with the proper kernel module params
202 203 204 205
	for dev in $rootdev $pvs; do
		[ -e "/sys/block/${dev#/dev/}/md" ] || continue

		local md=${dev#/dev/}
206
		initfs_features="${initfs_features% raid} raid"
207 208
		local level=$(cat /sys/block/$md/md/level)
		case "$level" in
209 210
			raid1) raidmod="${raidmod%,raid1},raid1";;
			raid[456]) raidmod="${raidmod%,raid456},raid456";;
211 212
		esac
	done
213 214 215 216

	# check if we need hardware raid drivers
	case $rootdev in
		/dev/cciss/*)
217
			initfs_features="${initfs_features% raid} raid"
218 219 220
			;;
	esac

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

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

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

244 245 246 247 248 249 250 251 252 253
	# generate update-extlinux.conf
	root=$(uuid_or_device $rootdev)
	if is_vmware; then
		kernel_opts="pax_nouderef $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
254 255 256 257
	if [ "$(rc --sys)" = "XEN0" ]; then
		sed -i -e "s:^default=.*:default=xen-grsec:" \
			"$mnt"/etc/update-extlinux.conf
	fi
258

259 260 261 262 263 264 265 266 267 268 269
	# 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

270
	# remove the installed db in case its there so we force re-install
271
	rm -f "$mnt"/var/lib/apk/installed "$mnt"/lib/apk/db/installed
272
	echo "Installing system on $rootdev:"
273 274 275 276
	# 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/

277

278
	local apkflags="--initdb --quiet --progress --update-cache --clean-protected"
279
	local pkgs=$(cat "$mnt"/etc/apk/world "$mnt"/var/lib/apk/world 2>/dev/null)
280
	pkgs="$pkgs acct linux-$KERNEL_FLAVOR alpine-base"
281 282 283
	if [ "$(rc --sys)" = "XEN0" ]; then
		pkgs="$pkgs xen-hypervisor"
	fi
284 285 286 287 288
	local repos=$(sed -e 's/\#.*//' /etc/apk/repositories)
	local repoflags=
	for i in $repos; do
		repoflags="$repoflags --repository $i"
	done
289 290
	
	apk add --root "$mnt" $apkflags --overlay-from-stdin \
291
		$repoflags $pkgs <$ovlfiles>/dev/null || return 1
292
	echo ""
293
	extlinux $extlinux_raidopt --install "$mnt"/boot
294 295 296 297
}

unmount_partitions() {
	local mnt="$1"
298 299

	# unmount the partitions
300
	umount $(awk '{print $2}' /proc/mounts | egrep "^$mnt(/|\$)" | sort -r)
301
}
302

303 304 305
# figure out decent default swap size in mega bytes
find_swap_size() {
	local memtotal_kb=$(awk '$1 == "MemTotal:" {print $2}' /proc/meminfo)
306 307 308 309
	# 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
310 311
		local sysfsdev=$(echo ${disk#/dev/} | sed 's:/:!:g')
		local sysfspath=/sys/block/$sysfsdev/size
312 313 314 315 316 317 318 319 320 321 322 323 324 325 326
		# 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
327 328 329 330
}

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

Natanael Copa's avatar
Natanael Copa committed
341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360
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
361 362 363 364
	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
365 366 367 368 369 370 371

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

	return 0
}

372
find_disks() {
Natanael Copa's avatar
Natanael Copa committed
373
	local p=
374
	for p in $(awk '$1 ~ /[0-9]+/ {print $4}' /proc/partitions); do
Natanael Copa's avatar
Natanael Copa committed
375
		is_available_disk $p && echo -n " $p"
376 377 378
	done
}

379 380 381 382 383 384 385 386 387 388 389
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"
390
	apk add -q sfdisk e2fsprogs lvm2 $raidpkg syslinux $@
391 392
}

393
show_disk_info() {
Natanael Copa's avatar
Natanael Copa committed
394
	local disk= vendor= model= d= size=
395
	for disk in $@; do
396 397
		local dev=${disk#/dev/}
		d=$(echo $dev | sed 's:/:!:g')
398 399 400
		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)
401
		echo "  $dev	($size $vendor $model)"
402 403
	done
}
404

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

418
# setup partitions on given disk dev in $1.
419 420 421 422 423 424
# usage: setup_partitions <diskdev> size1,type1 [size2,type2 ...] 
setup_partitions() {
	local diskdev="$1"
	shift

	echo "Initializing partitions on $diskdev..."
425 426 427 428 429 430 431

	# 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

432 433 434
	# fix the MBR while here so extlinux can boot
	cat "$MBR" > $diskdev

435 436
	local start=0
	local line=
437 438
	# create new partitions
	(
439 440 441 442
		for line in "$@"; do
			echo "$start,$line"
			start=
		done
443
	) | sfdisk -q -L -uM $diskdev >/dev/null || return 1
444 445 446 447 448 449 450 451 452 453 454

	# 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
455
# this is not marked as bootable and is type 8e
456 457 458 459 460
find_lvm_partition() {
	local type=8e
	sfdisk -d $1 | grep -v bootable | awk "/Id=$type/ {print \$1}"
}

461
# set up optional raid and create filesystem on boot device.
462
setup_boot_dev() {
463 464 465 466
	local disk= bootdev=
	local part=$(for disk in $@; do find_boot_partition $disk; done)
	set -- $part
	bootdev=$1
467
	[ -z "$bootdev" ] && return 1
468
	echo "Creating file systems..."
469
	if [ -n "$USE_RAID" ]; then
470 471 472 473 474 475
		local missing=
		local num=$#
		if [ $# -eq 1 ]; then
			missing="missing"
			num=2
		fi
476
		# we only use raid level 1 for boot devices
477 478
		mdadm --create /dev/md0 --level=1 --raid-devices=$num \
			--metadata=0.90 --quiet --run $@ $missing || return 1
479 480
		bootdev=/dev/md0
	fi
481
	mkfs.$BOOTFS -q $bootdev
482 483 484 485
	BOOT_DEV="$bootdev"
}

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

setup_non_boot_raid_dev() {
	local md_dev=$1
	local idx=${md_dev#/dev/md}
	shift
	local level=1
	local missing=
506
	local raid_parts=$(find_nth_non_boot_parts $idx "fd" $@)
507 508 509 510 511 512 513 514 515 516
	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
517 518 519 520
}

# setup device for lvm, create raid array if needed
setup_lvm_volume_group() {
521 522 523
	local vgname="$1"
	shift
	local lvmdev=
524 525

	if [ -n "$USE_RAID" ]; then
526 527 528 529
		setup_non_boot_raid_dev /dev/md1 $@ || return 1
		lvmdev=/dev/md1
	else	
		lvmdev=$(find_lvm_partition $1)
530 531 532 533 534
	fi

	# be quiet on success
	local errmsg=$(dd if=/dev/zero of=$lvmdev bs=1k count=1 2>&1) \
		|| echo "$errmsg"
535 536 537 538 539 540 541 542 543 544 545 546 547 548
	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
549 550 551
}

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

# 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
}

570 571 572
# set up /var on given device
setup_var() {
	local var_dev="$1"
573
	local varfs=${VARFS}
574
	echo "Creating file systems..."
575 576 577 578 579 580 581 582 583 584 585
	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
586
	setup_mdadm_conf
587 588
}

589
setup_mdadm_conf() {
590
	local mods= mod=
591 592 593
	if [ -n "$USE_RAID" ]; then
		mdadm --detail --scan > /etc/mdadm.conf
		rc-update --quiet add mdadm-raid boot
594 595 596 597 598 599
		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
600 601 602
	fi
}

603
data_only_disk_install_lvm() {
604
	local diskdev=
605 606
	local vgname=vg0
	local var_dev=/dev/$vgname/lv_var
607 608 609
	local lvm_part_type="8e"
	local part_type=$lvm_part_type
	local size=
610 611

	init_progs || return 1
612
	confirm_erase $@ || return 1
613

614
	if [ "$USE_RAID" ]; then
615 616
		# the paritition type for raid is "fd"
		part_type="fd"
617 618
		stop_all_raid
	fi
619

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

632 633
data_only_disk_install() {
	local diskdev=
634 635
	local var_dev=
	local var_part_type="83"
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 662 663 664 665 666 667
	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
}

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

	setup_mdadm_conf
680 681
	install_mounted_root "$SYSROOT" || return 1
	unmount_partitions "$SYSROOT"
682 683 684
	swapoff -a

	echo ""
685
	echo "Installation is complete. Please reboot."
686 687
}

688
native_disk_install_lvm() {
689 690 691 692 693 694 695
	local diskdev= vgname=vg0
	local lvm_part_type="8e"
	local boot_part_type="83"
	local boot_size=${BOOT_SIZE:-100}
	local lvm_size=
	local root_dev=/dev/$vgname/lv_root

696
	init_progs || return 1
697
	confirm_erase $@ || return 1
698 699

	if [ -n "$USE_RAID" ]; then
700 701 702
		boot_part_type="fd"
		lvm_part_type="fd"
		stop_all_raid
703
	fi
704 705 706 707 708 709 710 711 712 713 714 715
	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
716
	rc-update add lvm boot
717
	setup_root $root_dev $BOOT_DEV
Natanael Copa's avatar
Natanael Copa committed
718 719
}

720 721 722 723 724 725 726
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=

727
	init_progs || return 1
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 755 756 757 758 759
	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
}

760
diskselect_help() {
761 762
	cat <<__EOF__

763
The disk you select can be used for a traditional disk install or for a
764 765 766 767
data-only install.

The disk will be erased.

768
Enter 'none' if you want to run diskless.
769 770 771 772

__EOF__
}

773
diskmode_help() {
774 775
	cat <<__EOF__

776
You can select between 'sys' or 'data'.
777

778 779 780 781 782
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.
783

784 785 786
data:
  This mode uses your disk(s) for data storage, not for the operating system.
  The system itself will run from tmpfs (RAM).
787

788 789
  Use this mode if you only want to use the disk(s) for a mailspool, databases,
  logs, etc.
790 791 792 793 794 795 796 797 798

__EOF__
}

# ask for a root or data disk
# returns answer in global variable $answer
ask_disk() {
	local prompt="$1"
	local help_func="$2"
799
	local i=
800 801
	shift 2
	answer=
802
	local default_disk=${DEFAULT_DISK:-$1}
803

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

823 824
usage() {
	cat <<__EOF__
825
usage: setup-disk [-hLqrv] [-k kernelflavor] [-m MODE] [-o apkovl] [-s SWAPSIZE]
826 827 828 829 830 831 832 833 834 835 836 837 838
		  [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
839
 -m  Use disk for MODE without asking, where MODE is either 'data' or 'sys'
840 841
 -o  Restore system from given apkovl file
 -k  Use kernelflavor instead of $KERNEL_FLAVOR
842
 -L  Use LVM to manage partitions
843 844
 -q  Exit quietly if no disks are found
 -r  Enable software raid1 with single disk
845
 -s  Use SWAPSIZE MB instead of autodetecting swap size (Use 0 to disable swap)
846
 -v  Be more verbose about what is happening
847

848 849 850 851
__EOF__
	exit 1
}

852 853 854
KERNEL_FLAVOR=grsec
case "$(uname -r)" in
	*-vs[0-9]*) KERNEL_FLAVOR=vserver;;
855
	*-pae) KERNEL_FLAVOR=pae;;
856 857
esac

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

876 877
if [ -d "$1" ]; then
	# install to given mounted root
878
	apk add -q syslinux
879 880
	install_mounted_root "${1%/}" \
		&& echo "You might need fix the MBR to be able to boot" >&2
881 882 883
	exit $?
fi

884 885 886 887 888 889 890 891 892 893
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

894
disks=$(find_disks)
895
diskdevs=
Natanael Copa's avatar
Natanael Copa committed
896 897

# no disks so lets exit quietly.
898 899 900 901
if [ -z "$disks" ]; then
	[ -z "$QUIET" ] && echo "No disks found." >&2
	exit 0
fi
Natanael Copa's avatar
Natanael Copa committed
902

903 904 905 906 907
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
908
			echo "$i is not a suitable for partitioning"
909 910
			exit 1
		fi
911
		diskdevs="$diskdevs /dev/$j"
912 913
	done
else
914
	ask_disk "Which disk(s) would you like to use? (or '?' for help or 'none')" \
915
		diskselect_help $disks
916
	if [ "$answer" != none ]; then
917 918 919
		for i in $answer; do
			diskdevs="$diskdevs /dev/$i"
		done
920 921
	else
		DISK_MODE="none"
922
	fi
923
fi
Natanael Copa's avatar
Natanael Copa committed
924

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

948 949 950 951
if [ -z "$SWAP_SIZE" ]; then
	SWAP_SIZE=$(find_swap_size $diskdevs)
fi

952 953 954 955
set -- $diskdevs
if [ $# -gt 1 ]; then
	USE_RAID=1
fi
956

957 958
dmesg -n1

959
# native disk install
960
case "$DISK_MODE" in
961
sys) native_disk_install$USE_LVM $diskdevs;;
962
data) data_only_disk_install$USE_LVM $diskdevs;;
963
none) exit 0;;
964
*) echo "Not a valid install mode: $DISK_MODE" >&2; exit 1;;
965
esac
Natanael Copa's avatar
Natanael Copa committed
966

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