#!/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"