| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574 | #!/usr/bin/env bash#-# Copyright © 2010-2014#	Waldemar Brodkorb <wbx@openadk.org>#	Thorsten Glaser <tg@mirbsd.org>## Provided that these terms and disclaimer and all copyright notices# are retained or reproduced in an accompanying document, permission# is granted to deal in this work without restriction, including un‐# limited rights to use, publicly perform, distribute, sell, modify,# merge, give away, or sublicence.## This work is provided “AS IS” and WITHOUT WARRANTY of any kind, to# the utmost extent permitted by applicable law, neither express nor# implied; without malicious intent or gross negligence. In no event# may a licensor, author or contributor be held liable for indirect,# direct, other damage, loss, or other issues arising in any way out# of dealing in the work, even if advised of the possibility of such# damage or existence of a defect, except proven that it results out# of said person’s immediate fault when using the work as intended.## Alternatively, this work may be distributed under the terms of the# General Public License, any version, as published by the Free Soft-# ware Foundation.#-# Prepare a USB stick or CF/SD/MMC card or hard disc for installation# of OpenADKADK_TOPDIR=$(pwd)HOST=$(gcc -dumpmachine)me=$0case :$PATH: in(*:$ADK_TOPDIR/host_$HOST/usr/bin:*) ;;(*) export PATH=$PATH:$ADK_TOPDIR/host_$HOST/usr/bin ;;esactest -n "$KSH_VERSION" || exec mksh "$me" "$@"if test -z "$KSH_VERSION"; then	echo >&2 Fatal error: could not run myself with mksh!	exit 255fi### run with mksh from here onwards ###me=${me##*/}if (( USER_ID )); then	print -u2 Installation is only possible as root!	exit 1fiADK_TOPDIR=$(realpath .)ostype=$(uname -s)fs=ext4cfgfs=1datafssz=0noformat=0quiet=0serial=0speed=115200panicreboot=10function usage {cat >&2 <<EOFSyntax: $me [-f filesystem] [-c cfgfssize] [-d datafssize] [-n]    [-p panictime] [±q] [-s serialspeed] [±t] <target> <device> <archive>Partition sizes are in MiB. Filesystem type is currently ignored (ext4).Defaults: -c 1 -p 10 -s 115200; -t = enable serial consoleEOF	exit $1}while getopts "c:d:f:hnp:qs:t" ch; do	case $ch {	(c)	if (( (cfgfs = OPTARG) < 0 || cfgfs > 16 )); then			print -u2 "$me: -c $OPTARG out of bounds"			exit 1		fi ;;	(d)	if (( (datafssz = OPTARG) < 0 )); then			print -u2 "$me: -d $OPTARG out of bounds"			exit 1		fi ;;	(f)	if [[ $OPTARG != @(ext2|ext3|ext4|xfs) ]]; then			print -u2 "$me: filesystem $OPTARG invalid"			exit 1		fi		fs=$OPTARG ;;	(h)	usage 0 ;;	(p)	if (( (panicreboot = OPTARG) < 0 || panicreboot > 300 )); then			print -u2 "$me: -p $OPTARG out of bounds"			exit 1		fi ;;	(q)	quiet=1 ;;	(+q)	quiet=0 ;;	(s)	if [[ $OPTARG != @(96|192|384|576|1152)00 ]]; then			print -u2 "$me: serial speed $OPTARG invalid"			exit 1		fi		speed=$OPTARG ;;	(n)	noformat=1 ;;	(t)	serial=1 ;;	(+t)	serial=0 ;;	(*)	usage 1 ;;	}doneshift $((OPTIND - 1))(( $# == 3 )) || usage 1f=0case $ostype {(Linux)	tools="bc mkfs.$fs tune2fs"	;;(Darwin)	tools="bc diskutil"	;;(*)	print -u2 Sorry, not ported to the OS "'$ostype'" yet.	exit 1	;;}for tool in $tools; do	print -n Checking if $tool is installed...	if whence -p $tool >/dev/null; then		print " okay"	else		print " failed"		f=1	fidone(( f )) && exit 1target=$1tgt=$2src=$3case $target {(raspberry-pi|solidrun-imx6|default) ;;(*)	print -u2 "Unknown target '$target', exiting"	exit 1 ;;}if [[ ! -b $tgt ]]; then	print -u2 "'$tgt' is not a block device, exiting"	exit 1fiif [[ ! -f $src ]]; then	print -u2 "'$src' is not a file, exiting"	exit 1fi(( quiet )) || print "Installing $src on $tgt."case $ostype {(Darwin)	R=/Volumes/ADKROOT; diskutil unmount $R	B=/Volumes/ADKBOOT; diskutil unmount $B	D=/Volumes/ADKDATA; diskutil unmount $D	basedev=$tgt	rootpart=${basedev}s1	datapart=${basedev}s2	if [[ $target = raspberry-pi ]]; then		bootpart=${basedev}s1		rootpart=${basedev}s2		datapart=${basedev}s3	fi	match=\'${basedev}\''?(s+([0-9]))'	function mount_fs {	}	function umount_fs {		diskutil unmount "$1"	}	function create_fs {		if [[ $3 = ext4 ]]; then			fstype=UFSD_EXTFS4		fi		if [[ $3 = vfat ]]; then			fstype=fat32		fi		diskutil eraseVolume $fstype "$2" "$1"	}	;;(Linux)	basedev=$tgt	rootpart=${basedev}1	datapart=${basedev}2	if [[ $target = raspberry-pi ]]; then		bootpart=${basedev}1		rootpart=${basedev}2		datapart=${basedev}3	fi	match=\'${basedev}\''+([0-9])'	function mount_fs {		mount -t "$3" "$1" "$2"	}	function umount_fs {		umount "$1"	}	function create_fs {		mkfs.$3 "$1"		tune2fs -c 0 -i 0 "$1"	}	;;}mount |&while read -p dev rest; do	eval [[ \$dev = $match ]] || continue	print -u2 "Block device $tgt is in use, please umount first."	exit 1doneif (( !quiet )); then	print "WARNING: This will overwrite $basedev - type Yes to continue!"	read x	[[ $x = Yes ]] || exit 0fiif ! T=$(mktemp -d /tmp/openadk.XXXXXXXXXX); then	print -u2 Error creating temporary directory.	exit 1fiif [[ $ostype != Darwin ]]; then	R=$T/rootmnt	B=$T/bootmnt	D=$T/datamnt	mkdir -p "$R" "$B" "$D"fi# get disk sizedksz=$(dkgetsz "$tgt")# partition layouts:# n̲a̲m̲e̲		p̲a̲r̲t̲#̲0̲		p̲̲a̲r̲t̲#̲1̲		p̲̲a̲r̲t̲#̲2̲		p̲̲a̲r̲t̲#̲3̲# default:	0x83(system)	0x83(?data)	-(unused)	0x88(cfgfs)# raspberry:	0x0B(boot)	0x83(system)	0x83(?data)	0x88(cfgfs)syspartno=0# sizes:# boot(raspberry) - fixed (100 MiB)# cfgfs - fixed (parameter, max. 16 MiB)# data - flexible (parameter)# system - everything elseif [[ $target = raspberry-pi ]]; then	syspartno=1	bootfssz=100	if (( grub )); then		print -u2 "Cannot combine GRUB with $target"		rm -rf "$T"		exit 1	fielse	bootfssz=0fiheads=64secs=32(( cyls = dksz / heads / secs ))if (( cyls < (bootfssz + cfgfs + datafssz + 2) )); then	print -u2 "Size of $tgt is $dksz, this looks fishy?"	rm -rf "$T"	exit 1fiif stat -qs .>/dev/null 2>&1; then	statcmd='stat -f %z'	# BSD stat (or so we assume)else	statcmd='stat -c %s'	# GNU statfiif (( grub )); then	tar -xOzf "$src" boot/grub/core.img >"$T/core.img"	integer coreimgsz=$($statcmd "$T/core.img")else	coreimgsz=65024fiif (( coreimgsz < 1024 )); then	print -u2 core.img is probably too small: $coreimgsz	rm -rf "$T"	exit 1fiif (( coreimgsz > 65024 )); then	print -u2 core.img is larger than 64K-512: $coreimgsz	rm -rf "$T"	exit 1fi(( coreendsec = (coreimgsz + 511) / 512 ))if [[ $basedev = /dev/svnd+([0-9]) ]]; then	# BSD svnd0 mode: protect sector #1	corestartsec=2	(( ++coreendsec ))	corepatchofs=$((0x614))else	corestartsec=1	corepatchofs=$((0x414))fi# partition offset: at least coreendsec+1 but aligned on a multiple of secs#(( partofs = ((coreendsec / secs) + 1) * secs ))# we just use 2048 all the time, since some loaders are longerpartofs=2048if [[ $target = raspberry-pi ]]; then	(( spartofs = partofs + (100 * 2048) ))else	spartofs=$partofsfi(( quiet )) || if (( grub )); then	print Preparing MBR and GRUB2...else	print Preparing MBR...fidd if=/dev/zero of="$T/firsttrack" count=$partofs 2>/dev/null# add another MiB to clear the first partitiondd if=/dev/zero bs=1048576 count=1 >>"$T/firsttrack" 2>/dev/nullecho $corestartsec $coreendsec | mksh "$ADK_TOPDIR/scripts/bootgrub.mksh" \    -A -g $((cyls - bootfssz - cfgfs - datafssz)):$heads:$secs -M 1:0x83 \    -O $spartofs | dd of="$T/firsttrack" conv=notrunc 2>/dev/null(( grub )) && dd if="$T/core.img" of="$T/firsttrack" conv=notrunc \    seek=$corestartsec 2>/dev/null# set partition where it can find /boot/grub(( grub )) && print -n '\0\0\0\0' | \    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$corepatchofs 2>/dev/null# create cfgfs partition (mostly taken from bootgrub.mksh)set -A thecodetypeset -Uui8 thecodembrpno=0set -A g_code $cyls $heads $secs(( psz = g_code[0] * g_code[1] * g_code[2] ))(( pofs = (cyls - cfgfs) * g_code[1] * g_code[2] ))set -A o_code	# g_code equivalent for partition offset(( o_code[2] = pofs % g_code[2] + 1 ))(( o_code[1] = pofs / g_code[2] ))(( o_code[0] = o_code[1] / g_code[1] + 1 ))(( o_code[1] = o_code[1] % g_code[1] + 1 ))# boot flag; C/H/S offsetthecode[mbrpno++]=0x00(( thecode[mbrpno++] = o_code[1] - 1 ))(( cylno = o_code[0] > 1024 ? 1023 : o_code[0] - 1 ))(( thecode[mbrpno++] = o_code[2] | ((cylno & 0x0300) >> 2) ))(( thecode[mbrpno++] = cylno & 0x00FF ))# partition type; C/H/S end(( thecode[mbrpno++] = 0x88 ))(( thecode[mbrpno++] = g_code[1] - 1 ))(( cylno = g_code[0] > 1024 ? 1023 : g_code[0] - 1 ))(( thecode[mbrpno++] = g_code[2] | ((cylno & 0x0300) >> 2) ))(( thecode[mbrpno++] = cylno & 0x00FF ))# partition offset, size (LBA)(( thecode[mbrpno++] = pofs & 0xFF ))(( thecode[mbrpno++] = (pofs >> 8) & 0xFF ))(( thecode[mbrpno++] = (pofs >> 16) & 0xFF ))(( thecode[mbrpno++] = (pofs >> 24) & 0xFF ))(( pssz = psz - pofs ))(( thecode[mbrpno++] = pssz & 0xFF ))(( thecode[mbrpno++] = (pssz >> 8) & 0xFF ))(( thecode[mbrpno++] = (pssz >> 16) & 0xFF ))(( thecode[mbrpno++] = (pssz >> 24) & 0xFF ))# write partition table entryostr=curptr=0while (( curptr < 16 )); do	ostr=$ostr\\0${thecode[curptr++]#8#}doneprint -n "$ostr" | \    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1EE)) 2>/dev/nullif (( datafssz )); then	# create data partition (copy of the above :)	set -A thecode	typeset -Uui8 thecode	mbrpno=0	set -A g_code $cyls $heads $secs	(( psz = (cyls - cfgfs) * g_code[1] * g_code[2] ))	(( pofs = (cyls - cfgfs - datafssz) * g_code[1] * g_code[2] ))	set -A o_code	# g_code equivalent for partition offset	(( o_code[2] = pofs % g_code[2] + 1 ))	(( o_code[1] = pofs / g_code[2] ))	(( o_code[0] = o_code[1] / g_code[1] + 1 ))	(( o_code[1] = o_code[1] % g_code[1] + 1 ))	# boot flag; C/H/S offset	thecode[mbrpno++]=0x00	(( thecode[mbrpno++] = o_code[1] - 1 ))	(( cylno = o_code[0] > 1024 ? 1023 : o_code[0] - 1 ))	(( thecode[mbrpno++] = o_code[2] | ((cylno & 0x0300) >> 2) ))	(( thecode[mbrpno++] = cylno & 0x00FF ))	# partition type; C/H/S end	(( thecode[mbrpno++] = 0x83 ))	(( thecode[mbrpno++] = g_code[1] - 1 ))	(( cylno = (g_code[0] - cfgfs) > 1024 ? 1023 : g_code[0] - cfgfs - 1 ))	(( thecode[mbrpno++] = g_code[2] | ((cylno & 0x0300) >> 2) ))	(( thecode[mbrpno++] = cylno & 0x00FF ))	# partition offset, size (LBA)	(( thecode[mbrpno++] = pofs & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 8) & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 16) & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 24) & 0xFF ))	(( pssz = psz - pofs ))	(( thecode[mbrpno++] = pssz & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 8) & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 16) & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 24) & 0xFF ))	# write partition table entry	ostr=	curptr=0	while (( curptr < 16 )); do		ostr=$ostr\\0${thecode[curptr++]#8#}	done	print -n "$ostr" | \	    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1CE)) 2>/dev/nullfiif [[ $target = raspberry-pi ]]; then	# move system and data partition from #0/#1 to #1/#2	dd if="$T/firsttrack" bs=1 skip=$((0x1BE)) count=32 of="$T/x" 2>/dev/null	dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1CE)) if="$T/x" 2>/dev/null	# create boot partition (copy of the above :)	set -A thecode	typeset -Uui8 thecode	mbrpno=0	set -A g_code $cyls $heads $secs	psz=$spartofs	pofs=$partofs	set -A o_code	# g_code equivalent for partition offset	(( o_code[2] = pofs % g_code[2] + 1 ))	(( o_code[1] = pofs / g_code[2] ))	(( o_code[0] = o_code[1] / g_code[1] + 1 ))	(( o_code[1] = o_code[1] % g_code[1] + 1 ))	# boot flag; C/H/S offset	thecode[mbrpno++]=0x00	(( thecode[mbrpno++] = o_code[1] - 1 ))	(( cylno = o_code[0] > 1024 ? 1023 : o_code[0] - 1 ))	(( thecode[mbrpno++] = o_code[2] | ((cylno & 0x0300) >> 2) ))	(( thecode[mbrpno++] = cylno & 0x00FF ))	# partition type; C/H/S end	(( thecode[mbrpno++] = 0x0B ))	(( thecode[mbrpno++] = g_code[1] - 1 ))	(( cylno = (spartofs / 2048) > 1024 ? 1023 : (spartofs / 2048) - 1 ))	(( thecode[mbrpno++] = g_code[2] | ((cylno & 0x0300) >> 2) ))	(( thecode[mbrpno++] = cylno & 0x00FF ))	# partition offset, size (LBA)	(( thecode[mbrpno++] = pofs & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 8) & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 16) & 0xFF ))	(( thecode[mbrpno++] = (pofs >> 24) & 0xFF ))	(( pssz = psz - pofs ))	(( thecode[mbrpno++] = pssz & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 8) & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 16) & 0xFF ))	(( thecode[mbrpno++] = (pssz >> 24) & 0xFF ))	# write partition table entry	ostr=	curptr=0	while (( curptr < 16 )); do		ostr=$ostr\\0${thecode[curptr++]#8#}	done	print -n "$ostr" | \	    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1BE)) 2>/dev/nullfi# disk signaturernddev=/dev/urandom[[ -c /dev/arandom ]] && rnddev=/dev/arandomdd if=$rnddev bs=4 count=1 2>/dev/null | \    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1B8)) 2>/dev/nullprint -n '\0\0' | \    dd of="$T/firsttrack" conv=notrunc bs=1 seek=$((0x1BC)) 2>/dev/nullpartuuid=$(dd if="$T/firsttrack" bs=1 count=4 skip=$((0x1B8)) 2>/dev/null | \    hexdump -e '1/4 "%08x"')-0$((syspartno+1))(( quiet )) || print Cleaning out partitions...(( datafssz )) && dd if=/dev/zero bs=1048576 count=1 \    seek=$((cyls - cfgfs - datafssz)) 2>/dev/nulldd if=/dev/zero bs=1048576 count=1 seek=$((spartofs / 2048)) 2>/dev/null(( quiet )) || if (( grub )); then	print Writing MBR and GRUB2 to target device... system PARTUUID=$partuuidelse	print Writing MBR to target device... system PARTUUID=$partuuidfidd if="$T/firsttrack" of="$tgt"case $target {(solidrun-imx6)	fwdir=$(dirname "$src")	dd if="$fwdir/SPL" of="$tgt" bs=1024 seek=1	dd if="$fwdir/u-boot.img" of="$tgt" bs=1024 seek=42	;;(raspberry-pi)	(( quiet )) || print "Creating filesystem on ${bootpart}..."	(( noformat )) || create_fs "$bootpart" ADKBOOT vfat	;;}(( quiet )) || print "Creating filesystem on ${rootpart}..."(( noformat )) || create_fs "$rootpart" ADKROOT ext4if (( datafssz )); then	(( quiet )) || print "Creating filesystem on ${datapart}..."	(( noformat )) || create_fs "$datapart" ADKDATA ext4	mount_fs "$datapart" "$D" ext4	mkdir -m0755 "$D/mpd" "$D/xbmc"	umount_fs "$D"fi(( quiet )) || print Extracting installation archive...mount_fs "$rootpart" "$R" ext4gzip -dc "$src" | (cd "$R"; tar -xpf -)case $target {(raspberry-pi)	mount_fs "$bootpart" "$B" vfat	for x in "$R"/boot/.*; do		[[ -e "$x" ]] && mv -f "$R"/boot/.* "$B/"		break	done	for x in "$R"/boot/*; do		[[ -e "$x" ]] && mv -f "$R"/boot/* "$B/"		break	done	umount_fs "$B"	;;(solidrun-imx6)	for x in "$fwdir"/*.dtb; do		[[ -e "$x" ]] && cp "$fwdir"/*.dtb "$R/boot/"		break	done	;;}cd "$R"dd if=$rnddev bs=16 count=1 >>etc/.rnd 2>/dev/null(( quiet )) || print Fixing up permissions...chown 0:0 tmpchmod 1777 tmp[[ -f usr/bin/sudo ]] && chmod 4755 usr/bin/sudoif (( grub )); then	(( quiet )) || print Configuring GRUB2 bootloader...	mkdir -p boot/grub	(		print set default=0		print set timeout=1		if (( serial )); then			print serial --unit=0 --speed=$speed			print terminal_output serial			print terminal_input serial			consargs="console=ttyS0,$speed console=tty0"		else			print terminal_output console			print terminal_input console			consargs="console=tty0"		fi		print		print 'menuentry "GNU/Linux (OpenADK)" {'		linuxargs="root=PARTUUID=$partuuid $consargs"		(( panicreboot )) && linuxargs="$linuxargs panic=$panicreboot"		print "\tlinux /boot/kernel $linuxargs"		print '}'	) >boot/grub/grub.cfgfi(( quiet )) || print Finishing up...cd "$ADK_TOPDIR"syncumount_fs "$R"syncrm -rf "$T"exit 0
 |