blob: fcc42741c6ea1f56d924d35f9dc7be2368173c98 [file]
#!/bin/bash
# Clear environment variables by restarting script w/bare minimum passed through
[ -z "$NOCLEAR" ] &&
exec env -i NOCLEAR=1 HOME="$HOME" PATH="$PATH" LINUX="$LINUX" \
CROSS_COMPILE="$CROSS_COMPILE" CROSS_SHORT="$CROSS_SHORT" "$0" "$@"
# assign command line NAME=VALUE args to env vars
while [ $# -ne 0 ]
do
X="${1/=*/}"
Y="${1#*=}"
[ "${1/=/}" != "$1" ] && eval "export $X=\"\$Y\"" || echo "unknown $i"
shift
done
# If we're cross compiling, set appropriate environment variables.
if [ -z "$CROSS_COMPILE" ]
then
echo "Building natively"
if ! cc --static -xc - -o /dev/null <<< "int main(void) {return 0;}"
then
echo "Warning: host compiler can't create static binaries." >&2
sleep 3
fi
else
CROSS_PATH="$(dirname "$(which "${CROSS_COMPILE}cc")")"
CROSS_BASE="$(basename "$CROSS_COMPILE")"
[ -z "$CROSS_SHORT" ] && CROSS_SHORT="${CROSS_BASE/-*/}"
echo "Cross compiling to $CROSS_SHORT"
if [ -z "$CROSS_PATH" ]
then
echo "no ${CROSS_COMPILE}cc in path" >&2
exit 1
fi
fi
# set up directories (can override most of these paths on cmdline)
TOP="$PWD/root"
[ -z "$BUILD" ] && BUILD="$TOP/build"
[ -z "$AIRLOCK" ] && AIRLOCK="$TOP/airlock"
[ -z "$OUTPUT" ] && OUTPUT="$TOP/${CROSS_SHORT:-host}"
[ -z "$ROOT" ] && ROOT="$OUTPUT/${CROSS_BASE}fs" && rm -rf "$ROOT"
MYBUILD="$BUILD/${CROSS_BASE:-host-}tmp"
rm -rf "$MYBUILD" && mkdir -p "$MYBUILD" || exit 1
# Stabilize cross compiling by providing known $PATH contents
if [ ! -z "$CROSS_COMPILE" ]
then
if [ ! -e "$AIRLOCK/toybox" ]
then
echo === Create airlock dir
PREFIX="$AIRLOCK" KCONFIG_CONFIG="$TOP"/.airlock CROSS_COMPILE= \
make clean defconfig toybox install_airlock &&
rm "$TOP"/.airlock || exit 1
fi
export PATH="$CROSS_PATH:$AIRLOCK"
fi
### Create files and directories
mkdir -p "$ROOT"/{etc,tmp,proc,sys,dev,home,mnt,root,usr/{bin,sbin,lib},var} &&
chmod a+rwxt "$ROOT"/tmp && ln -s usr/{bin,sbin,lib} "$ROOT" || exit 1
# init script. Runs as pid 1 from initramfs to set up and hand off system.
cat > "$ROOT"/init << 'EOF' &&
#!/bin/sh
export HOME=/home
export PATH=/bin:/sbin
mountpoint -q proc || mount -t proc proc proc
mountpoint -q sys || mount -t sysfs sys sys
if ! mountpoint -q dev
then
mount -t devtmpfs dev dev || mdev -s
mkdir -p dev/pts
mountpoint -q dev/pts || mount -t devpts dev/pts dev/pts
fi
if [ $$ -eq 1 ]
then
# Setup networking for QEMU (needs /proc)
ifconfig eth0 10.0.2.15
route add default gw 10.0.2.2
[ "$(date +%s)" -lt 1000 ] && rdate 10.0.2.2 # or time-b.nist.gov
[ "$(date +%s)" -lt 10000000 ] && ntpd -nq -p north-america.pool.ntp.org
[ -z "$CONSOLE" ] &&
CONSOLE="$(sed -n 's@.* console=\(/dev/\)*\([^ ]*\).*@\2@p' /proc/cmdline)"
[ -z "$HANDOFF" ] && HANDOFF=/bin/sh && echo Type exit when done.
[ -z "$CONSOLE" ] && CONSOLE=console
exec /sbin/oneit -c /dev/"$CONSOLE" $HANDOFF
else
/bin/sh
umount /dev/pts /dev /sys /proc
fi
EOF
chmod +x "$ROOT"/init &&
# /etc/passwd with both kernel special accounts (root and nobody) + guest user
cat > "$ROOT"/etc/passwd << 'EOF' &&
root::0:0:root:/home/root:/bin/sh
guest:x:500:500:guest:/home/guest:/bin/sh
nobody:x:65534:65534:nobody:/proc/self:/dev/null
EOF
# /etc/group with groups corresponding to each /etc/passwd user
cat > "$ROOT"/etc/group << 'EOF' &&
root:x:0:
guest:x:500:
nobody:x:65534:
EOF
# /etc/resolv.conf using Google's public nameserver. (We could use QEMU's
# 10.0.2.2 forwarder here, but this way works in both chroot and QEMU.)
echo "nameserver 8.8.8.8" > "$ROOT"/etc/resolv.conf || exit 1
# Build toybox
make clean
if [ -z .config ]
then
make defconfig
# Work around musl-libc design flaw.
[ "${CROSS_BASE/fdpic//}" != "$CROSS_BASE" ] &&
sed -i 's/.*\(CONFIG_TOYBOX_MUSL_NOMMU_IS_BROKEN\).*/\1=y/' .config
else
make silentoldconfig
fi
LDFLAGS=--static PREFIX="$ROOT" make toybox install || exit 1
# Abort early if no kernel source specified
if [ -z "$LINUX" ] || [ ! -d "$LINUX/kernel" ]
then
echo 'No $LINUX directory, kernel build skipped.'
rmdir "$MYBUILD" "$BUILD" 2>/dev/null
exit 0
fi
# Which architecture are we building a kernel for?
[ -z "$TARGET" ] && TARGET="${CROSS_BASE/-*/}"
[ -z "$TARGET" ] && TARGET="$(uname -m)"
# Target-specific info in an (alphabetical order) if/else staircase
# Each target needs board config, serial console, RTC, ethernet, block device.
if [ "$TARGET" == armv5l ]
then
# This could use the same VIRT board as armv7, but let's demonstrate a
# different one requiring a separate device tree binary.
QEMU="qemu-system-arm -M versatilepb -net nic,model=rtl8139 -net user"
KARCH=arm
KARGS="console=ttyAMA0"
VMLINUX=arch/arm/boot/zImage
KERNEL_CONFIG="
CONFIG_CPU_ARM926T=y
CONFIG_MMU=y
CONFIG_VFP=y
CONFIG_ARM_THUMB=y
CONFIG_AEABI=y
CONFIG_ARCH_VERSATILE=y
# The switch to device-tree-only added this mess
CONFIG_ATAGS=y
CONFIG_DEPRECATED_PARAM_STRUCT=y
CONFIG_ARM_ATAG_DTB_COMPAT=y
CONFIG_ARM_ATAG_DTB_COMPAT_CMDLINE_EXTEND=y
CONFIG_SERIAL_AMBA_PL011=y
CONFIG_SERIAL_AMBA_PL011_CONSOLE=y
CONFIG_RTC_CLASS=y
CONFIG_RTC_DRV_PL031=y
CONFIG_RTC_HCTOSYS=y
CONFIG_PCI=y
CONFIG_PCI_VERSATILE=y
CONFIG_BLK_DEV_SD=y
CONFIG_SCSI=y
CONFIG_SCSI_LOWLEVEL=y
CONFIG_SCSI_SYM53C8XX_2=y
CONFIG_SCSI_SYM53C8XX_DMA_ADDRESSING_MODE=0
CONFIG_SCSI_SYM53C8XX_MMIO=y
CONFIG_NET_VENDOR_REALTEK=y
CONFIG_8139CP=y
"
DTB=arch/arm/boot/dts/versatile-pb.dtb
elif [ "$TARGET" == armv7l ] || [ "$TARGET" == aarch64 ]
then
if [ "$TARGET" == aarch64 ]
then
QEMU="qemu-system-aarch64 -M virt -cpu cortex-a57"
KARCH=arm64
VMLINUX=arch/arm64/boot/Image
else
QEMU="qemu-system-arm -M virt"
KARCH=arm
VMLINUX=arch/arm/boot/zImage
fi
KARGS="console=ttyAMA0"
KERNEL_CONFIG="
CONFIG_MMU=y
CONFIG_ARCH_MULTI_V7=y
CONFIG_ARCH_VIRT=y
CONFIG_SOC_DRA7XX=y
CONFIG_ARCH_OMAP2PLUS_TYPICAL=y
CONFIG_ARCH_ALPINE=y
CONFIG_ARM_THUMB=y
CONFIG_VDSO=y
CONFIG_CPU_IDLE=y
CONFIG_ARM_CPUIDLE=y
CONFIG_KERNEL_MODE_NEON=y
CONFIG_SERIAL_AMBA_PL011=y
CONFIG_SERIAL_AMBA_PL011_CONSOLE=y
CONFIG_RTC_CLASS=y
CONFIG_RTC_HCTOSYS=y
CONFIG_RTC_DRV_PL031=y
CONFIG_NET_CORE=y
CONFIG_VIRTIO_MENU=y
CONFIG_VIRTIO_NET=y
CONFIG_PCI=y
CONFIG_PCI_HOST_GENERIC=y
CONFIG_VIRTIO_BLK=y
CONFIG_VIRTIO_PCI=y
CONFIG_VIRTIO_MMIO=y
CONFIG_ATA=y
CONFIG_ATA_SFF=y
CONFIG_ATA_BMDMA=y
CONFIG_ATA_PIIX=y
CONFIG_PATA_PLATFORM=y
CONFIG_PATA_OF_PLATFORM=y
CONFIG_ATA_GENERIC=y
"
elif [ "$TARGET" == i486 ] || [ "$TARGET" == i686 ] ||
[ "$TARGET" == x86_64 ] || [ "$TARGET" == x32 ]
then
if [ "$TARGET" == i486 ]
then
QEMU="qemu-system-i386 -cpu 486 -global fw_cfg.dma_enabled=false"
KERNEL_CONFIG="CONFIG_M486=y"
elif [ "$TARGET" == i686 ]
then
QEMU="qemu-system-i386 -cpu pentium3"
KERNEL_CONFIG="CONFIG_MPENTIUMII=y"
else
QEMU=qemu-system-x86_64
KERNEL_CONFIG="CONFIG_64BIT=y"
[ "$TARGET" == x32 ] && KERNEL_CONFIG="$KERNEL_CONFIG
CONFIG_X86_X32=y"
fi
KARCH=x86
KARGS="console=ttyS0"
VMLINUX=arch/x86/boot/bzImage
CONFIG_MPENTIUMII=y
KERNEL_CONFIG="
$KERNEL_CONFIG
CONFIG_UNWINDER_FRAME_POINTER=y
CONFIG_PCI=y
CONFIG_BLK_DEV_SD=y
CONFIG_ATA=y
CONFIG_ATA_SFF=y
CONFIG_ATA_BMDMA=y
CONFIG_ATA_PIIX=y
CONFIG_NET_VENDOR_INTEL=y
CONFIG_E1000=y
CONFIG_SERIAL_8250=y
CONFIG_SERIAL_8250_CONSOLE=y
CONFIG_RTC_CLASS=y
"
elif [ "$TARGET" == mips ] || [ "$TARGET" == mipsel ]
then
QEMU="qemu-system-mips -M malta"
KARCH=mips
KARGS="console=ttyS0"
VMLINUX=vmlinux
KERNEL_CONFIG="
CONFIG_MIPS_MALTA=y
CONFIG_CPU_MIPS32_R2=y
CONFIG_SERIAL_8250=y
CONFIG_SERIAL_8250_CONSOLE=y
CONFIG_PCI=y
CONFIG_BLK_DEV_SD=y
CONFIG_ATA=y
CONFIG_ATA_SFF=y
CONFIG_ATA_BMDMA=y
CONFIG_ATA_PIIX=y
CONFIG_NET_VENDOR_AMD=y
CONFIG_PCNET32=y
CONFIG_POWER_RESET=y
CONFIG_POWER_RESET_SYSCON=y
"
[ "$TARGET" == mipsel ] &&
KERNEL_CONFIG="${KERNEL_CONFIG}CONFIG_CPU_LITTLE_ENDIAN=y" &&
QEMU="qemu-system-mipsel -M malta"
elif [ "$TARGET" == powerpc ]
then
KARCH=powerpc
QEMU="qemu-system-ppc -M g3beige"
KARGS="console=ttyS0"
VMLINUX=vmlinux
KERNEL_CONFIG="
CONFIG_ALTIVEC=y
CONFIG_PPC_PMAC=y
CONFIG_PPC_OF_BOOT_TRAMPOLINE=y
CONFIG_IDE=y
CONFIG_IDE_GD=y
CONFIG_IDE_GD_ATA=y
CONFIG_BLK_DEV_IDE_PMAC=y
CONFIG_BLK_DEV_IDE_PMAC_ATA100FIRST=y
CONFIG_MACINTOSH_DRIVERS=y
CONFIG_ADB=y
CONFIG_ADB_CUDA=y
CONFIG_NET_VENDOR_NATSEMI=y
CONFIG_NET_VENDOR_8390=y
CONFIG_NE2K_PCI=y
CONFIG_SERIO=y
CONFIG_SERIAL_PMACZILOG=y
CONFIG_SERIAL_PMACZILOG_TTYS=y
CONFIG_SERIAL_PMACZILOG_CONSOLE=y
CONFIG_BOOTX_TEXT=y
"
elif [ "$TARGET" == powerpc64le ]
then
KARCH=powerpc
QEMU="qemu-system-ppc64 -M pseries -vga none"
KARGS="console=/dev/hvc0"
VMLINUX=vmlinux
KERNEL_CONFIG="CONFIG_PPC64=y
CONFIG_PPC_PSERIES=y
CONFIG_CPU_LITTLE_ENDIAN=y
CONFIG_PPC_OF_BOOT_TRAMPOLINE=y
CONFIG_BLK_DEV_SD=y
CONFIG_SCSI_LOWLEVEL=y
CONFIG_SCSI_IBMVSCSI=y
CONFIG_ATA=y
CONFIG_NET_VENDOR_IBM=y
CONFIG_IBMVETH=y
CONFIG_HVC_CONSOLE=y
# None of this should be necessary
CONFIG_PPC_TRANSACTIONAL_MEM=y
CONFIG_PPC_DISABLE_WERROR=y
CONFIG_SECTION_MISMATCH_WARN_ONLY=y
"
elif [ "$TARGET" = s390x ]
then
QEMU="qemu-system-s390x"
KARCH=s390
VMLINUX=arch/s390/boot/bzImage
KERNEL_CONFIG="
CONFIG_MARCH_Z900=y
CONFIG_PACK_STACK=y
CONFIG_NET_CORE=y
CONFIG_VIRTIO_NET=y
CONFIG_VIRTIO_BLK=y
CONFIG_SCLP_TTY=y
CONFIG_SCLP_CONSOLE=y
CONFIG_SCLP_VT220_TTY=y
CONFIG_SCLP_VT220_CONSOLE=y
CONFIG_S390_GUEST=y
"
elif [ "$TARGET" == sh4 ]
then
QEMU="qemu-system-sh4 -M r2d -serial null -serial mon:stdio"
KARCH=sh
KARGS="console=ttySC1 noiotrap"
VMLINUX=arch/sh/boot/zImage
KERNEL_CONFIG="
CONFIG_CPU_SUBTYPE_SH7751R=y
CONFIG_MMU=y
CONFIG_MEMORY_START=0x0c000000
CONFIG_VSYSCALL=y
CONFIG_SH_FPU=y
CONFIG_SH_RTS7751R2D=y
CONFIG_RTS7751R2D_PLUS=y
CONFIG_SERIAL_SH_SCI=y
CONFIG_SERIAL_SH_SCI_CONSOLE=y
CONFIG_PCI=y
CONFIG_NET_VENDOR_REALTEK=y
CONFIG_8139CP=y
CONFIG_PCI=y
CONFIG_BLK_DEV_SD=y
CONFIG_ATA=y
CONFIG_ATA_SFF=y
CONFIG_ATA_BMDMA=y
CONFIG_PATA_PLATFORM=y
CONFIG_BINFMT_ELF_FDPIC=y
CONFIG_BINFMT_FLAT=y
#CONFIG_SPI=y
#CONFIG_SPI_SH_SCI=y
#CONFIG_MFD_SM501=y
#CONFIG_RTC_CLASS=y
#CONFIG_RTC_DRV_R9701=y
#CONFIG_RTC_DRV_SH=y
#CONFIG_RTC_HCTOSYS=y
"
else
echo "Unknown \$TARGET"
exit 1
fi
# Write the miniconfig file
{
echo "# make ARCH=$KARCH allnoconfig KCONFIG_ALLCONFIG=$TARGET.miniconf"
echo "# make ARCH=$KARCH -j \$(nproc)"
echo "# boot $VMLINUX"
echo
echo "$KERNEL_CONFIG"
# Generic options for all targets
echo "
# CONFIG_EMBEDDED is not set
CONFIG_EARLY_PRINTK=y
CONFIG_BINFMT_ELF=y
CONFIG_BINFMT_SCRIPT=y
CONFIG_NO_HZ=y
CONFIG_HIGH_RES_TIMERS=y
CONFIG_BLK_DEV=y
CONFIG_BLK_DEV_INITRD=y
CONFIG_RD_GZIP=y
CONFIG_BLK_DEV_LOOP=y
CONFIG_EXT4_FS=y
CONFIG_EXT4_USE_FOR_EXT2=y
CONFIG_VFAT_FS=y
CONFIG_FAT_DEFAULT_UTF8=y
CONFIG_MISC_FILESYSTEMS=y
CONFIG_SQUASHFS=y
CONFIG_SQUASHFS_XATTR=y
CONFIG_SQUASHFS_ZLIB=y
CONFIG_DEVTMPFS=y
CONFIG_DEVTMPFS_MOUNT=y
CONFIG_TMPFS=y
CONFIG_TMPFS_POSIX_ACL=y
CONFIG_NET=y
CONFIG_PACKET=y
CONFIG_UNIX=y
CONFIG_INET=y
CONFIG_IPV6=y
CONFIG_NETDEVICES=y
#CONFIG_NET_CORE=y
#CONFIG_NETCONSOLE=y
CONFIG_ETHERNET=y
"
} > "$OUTPUT/miniconfig-$TARGET"
# Write the qemu launch script
echo "$QEMU -nographic -no-reboot -m 256" \
"-append \"panic=1 HOST=$TARGET $KARGS\"" \
"-kernel $(basename "$VMLINUX") -initrd ${CROSS_BASE}root.cpio.gz" \
${DTB:+-dtb "$(basename "$DTB")"} '"$@"' \
> "$OUTPUT/qemu-$TARGET.sh" &&
chmod +x "$OUTPUT/qemu-$TARGET.sh" &&
echo "Build linux for $KARCH"
# Snapshot Linux source dir and clean it
cp -sfR "$LINUX" "$MYBUILD/linux" && pushd "$MYBUILD/linux" > /dev/null ||
exit 1
# Build kernel
make distclean &&
make ARCH=$KARCH allnoconfig KCONFIG_ALLCONFIG="$OUTPUT/miniconfig-$TARGET" &&
make ARCH=$KARCH CROSS_COMPILE="$CROSS_COMPILE" -j $(nproc) || exit 1
# If we have a device tree binary, save it for QEMU.
if [ ! -z "$DTB" ]
then
cp "$DTB" "$OUTPUT/$(basename "$DTB")" || exit 1
fi
cp "$VMLINUX" "$OUTPUT/$(basename "$VMLINUX")" && cd .. && rm -rf linux &&
popd || exit 1
rmdir "$MYBUILD" "$BUILD" 2>/dev/null
# package root filesystem for initramfs.
# we do it here so module install can add files (not implemented yet)
echo === create "${CROSS_BASE}root.cpio.gz"
(cd "$ROOT" && find . | cpio -o -H newc | gzip) > \
"$OUTPUT/${CROSS_BASE}root.cpio.gz"