@ -1,11 +1,33 @@
####################################
# Makefile configuration variables #
OS_FILENAME = os.bat
BUILD_DIR = build
SCREENSHOTS_DIR = deploy-screenshots
COMMIT_TIMESTAMP_ISO_8601 = $$ ( git log -1 --pretty= format:%ad --date= iso8601-strict)
####################################
MAKEFLAGS = --warn-undefined-variables
SHELL = bash -euET -o pipefail -c
.SECONDEXPANSION :
os_filename = os.bat
os_filename = ${ OS_FILENAME }
bld = ${ BUILD_DIR }
screenshots = ${ SCREENSHOTS_DIR }
tests_emu = test/qemu-system-i386-floppy test/qemu-system-i386-cdrom test/qemu-system-arm test/virtualbox test/bochs test/gui-sh test/dosbox
tests_requiring_sudo = test/fat12_mount test/iso_mount
tests_noemu = test/zip test/os.reasm test/sizes test/fat12_contents
tests_noemu = test/zip test/os.reasm test/sizes test/fat12_contents test/reproducible_build
# We truncate the timezone, because the Darwin version of date seems to lack
# the %:z format (for ±HH:MM timezone).
d e f i n e d a t e _ c o m m a n d
if test " $$ (uname -s) " = Darwin; then \
date -j -f %Y-%m-%dT%H:%M:%S $$ ( echo ${ 1 } | cut -c 1-19) ${ 2 } ; \
else \
date -d ${ 1 } ${ 2 } ; \
fi
e n d e f
commit_timestamp = " $$ ( ${ call date_command, " ${ COMMIT_TIMESTAMP_ISO_8601 } " , '+%Y%m%d%H%m.%S' } ) "
commit_timestamp_iso_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 }
offset_names = bytes_os_size \
bytes_mbr_start \
@ -39,55 +61,72 @@ more_offset_names = ${offset_names} \
tracks_os_size \
tracks_zip_size
more_offset_dec = ${ more_offset_names : %=build/offsets/%.dec }
more_offset_hex = ${ more_offset_names : %=build/offsets/%.hex }
more_offset_dec = ${ more_offset_names : %= ${ bld } /offsets/%.dec }
more_offset_hex = ${ more_offset_names : %= ${ bld } /offsets/%.hex }
reproducible_os_filename = " ${ bld } /reproduced_ $$ (basename " ${ os_filename } ")"
# + os.arm.disasm
# + os.reasm.disasm
built_files = ${ os_filename } \
bui ld/check_makefile \
bui ld/check_makefile_targets \
bui ld/check_makefile_w_arnings \
bui ld/checkerboard_800x600.xbm \
bui ld/checkerboard_1024x768.png \
bui ld/makefile_built_directories \
bui ld/makefile_built_files \
bui ld/makefile_database \
bui ld/makefile_database_files \
bui ld/makefile_file_targets \
bui ld/makefile_non_file_targets \
bui ld/makefile_phony \
bui ld/makefile_targets \
bui ld/makefile_w_arnings \
bui ld/os.ndisasm.disasm \
bui ld/os.reasm.asm \
bui ld/os.reasm \
bui ld/os.file \
bui ld/os.gdisk \
bui ld/os.zip \
bui ld/os.zip.adjusted \
bui ld/os.iso \
bui ld/os.32k \
bui ld/os.fat12 \
bui ld/os.offsets.hex \
bui ld/os.offsets.dec \
bui ld/os.hex_with_offsets \
bui ld/iso_files/os.zip \
bui ld/iso_files/boot/iso_boot.sys \
bui ld/bochsrc \
bui ld/bochscontinue \
bui ld/twm_cfg \
bui ld/virtualbox.img \
${ bld} /check_makefile \
${ bld} /check_makefile_targets \
${ bld} /check_makefile_w_arnings \
${ bld} /checkerboard_800x600.xbm \
${ bld} /checkerboard_1024x768.png \
${ bld} /makefile_built_directories \
${ bld} /makefile_built_files \
${ bld} /makefile_database \
${ bld} /makefile_database_files \
${ bld} /makefile_file_targets \
${ bld} /makefile_non_file_targets \
${ bld} /makefile_phony \
${ bld} /makefile_targets \
${ bld} /makefile_w_arnings \
${ bld} /os.ndisasm.disasm \
${ bld} /os.reasm.asm \
${ bld} /os.reasm \
${ bld} /os.file \
${ bld} /os.gdisk \
${ bld} /os.zip \
${ bld} /os.zip.adjusted \
${ bld} /os.iso \
${ bld} /os.32k \
${ bld} /os.fat12 \
${ bld} /os.offsets.hex \
${ bld} /os.offsets.dec \
${ bld} /os.hex_with_offsets \
${ bld} /iso_files/os.zip \
${ bld} /iso_files/boot/iso_boot.sys \
${ bld} /bochsrc \
${ bld} /bochscontinue \
${ bld} /twm_cfg \
${ bld} /virtualbox.img \
${ more_offset_dec } \
${ more_offset_hex } \
${ tests_emu : test /%=build/test_pass/emu_% } \
${ tests_noemu : test /%=build/test_pass/noemu_% } \
${ tests_requiring_sudo : test /%=build/test_pass/sudo_% } \
${ tests_emu : test /%=deploy-screenshots/%.png } \
${ tests_emu : test /%=deploy-screenshots/%-anim.gif }
${ tests_emu : test /%= ${ bld } /test_pass/emu_% } \
${ tests_noemu : test /%= ${ bld } /test_pass/noemu_% } \
${ tests_requiring_sudo : test /%= ${ bld } /test_pass/sudo_% } \
${ tests_emu : test /%= ${ screenshots } /%.png } \
${ tests_emu : test /%= ${ screenshots } /%-anim.gif } \
utils/mformat utils/mcopy utils/mkisofs
built_directories = build/iso_files/boot build/iso_files build/offsets build/mnt_fat12 build/mnt_iso build/test_pass deploy-screenshots
more_built_directories = ${ built_directories } build
# Temporary copies used to adjust timestamps for reproducible builds.
# These are normally created and deleted within a single target, but
# could remain if make is interrupted during the build.
temp_files = ${ bld } /iso_files.tmp/os.zip \
${ bld } /iso_files.tmp/boot/iso_boot.sys
temp_directories = ${ bld } /iso_files.tmp \
${ bld } /iso_files.tmp/boot/
built_directories = ${ bld } /iso_files/boot \
${ bld } /iso_files \
${ bld } /offsets \
${ bld } /mnt_fat12 \
${ bld } /mnt_iso \
${ bld } /test_pass \
${ screenshots }
more_built_directories = ${ built_directories } ${ bld }
os_image_size_kb = 1440
os_partition_start_sectors = 3
@ -98,59 +137,92 @@ os_floppy_chs_s = 9
.PHONY : all
# all: os.arm.disasm
all : ${os_filename } build /os .ndisasm .disasm build /os .reasm .asm build /os .file build /os .gdisk build /os .offsets .hex build /os .offsets .dec build /os .hex_with_offsets .gitignore build /check_makefile ${more_offset_dec } ${more_offset_hex }
all : ${os_filename } \
${ bld } /os.ndisasm.disasm \
${ bld } /os.reasm.asm \
${ bld } /os.file \
${ bld } /os.gdisk \
${ bld } /os.offsets.hex \
${ bld } /os.offsets.dec \
${ bld } /os.hex_with_offsets \
.gitignore \
${ bld } /check_makefile \
${ more_offset_dec } \
${ more_offset_hex }
build/makefile_w_arnings : | $${@D }
${ bld} /makefile_w_arnings: | $${@D }
${built_files} : | $${@D }
build/makefile_w_arnings : Makefile
@unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; make -n --warn-undefined-variables test 2>$@ 1>/dev/null || make -n --warn-undefined-variables test
${bld}/makefile_w_arnings : Makefile
@unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make -n --warn-undefined-variables \
OS_FILENAME = ${ OS_FILENAME } \
BUILD_DIR = ${ BUILD_DIR } \
SCREENSHOTS_DIR = ${ SCREENSHOTS_DIR } \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
test 2>$@ 1>/dev/null \
|| cat $@
# Check that the file build/makefile_w_arnings is present, and that it does not contain the string "warn".
build/check_makefile_w_arnings : build /makefile_w_arnings
@cat build/makefile_w_arnings > /dev/null && ( ! grep -i warn $<) && touch $@
# Check that the file ${ bld} /makefile_w_arnings is present, and that it does not contain the string "warn".
${ bld} /check_makefile_w_arnings: ${ b ld} /makefile_w_arnings
@cat ${ bld} /makefile_w_arnings > /dev/null && ( ! grep -i warn $<) && touch $@
# Check that the declared list of built files matches the list of targets extracted from the Makefile.
build/check_makefile_targets : build /makefile_built_files build /makefile_file_targets build /check_makefile_w_arnings
@diff bui ld/makefile_built_files bui ld/makefile_file_targets && touch $@
${ bld} /check_makefile_targets: ${ b ld} /makefile_built_files ${ b ld} /makefile_file_targets ${ b ld} /check_makefile_w_arnings
@diff ${ bld} /makefile_built_files ${ bld} /makefile_file_targets && touch $@
bui ld/check_makefile: b ui ld/check_makefile_w_arnings b ui ld/check_makefile_targets
${ bld} /check_makefile: ${ b ld} /check_makefile_w_arnings ${ b ld} /check_makefile_targets
@touch $@
build/makefile_database : Makefile build /check_makefile_w_arnings
@unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; make -rpn | sed -n -e '/^# Make data base,/,$$p' > $@
${bld}/makefile_database : Makefile ${bld }/check_makefile_w_arnings
@unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make -rpn \
OS_FILENAME = ${ OS_FILENAME } \
BUILD_DIR = ${ BUILD_DIR } \
SCREENSHOTS_DIR = ${ SCREENSHOTS_DIR } \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
| sed -n -e '/^# Make data base,/,$$p' > $@
build/makefile_database_files : build /makefile_database build /check_makefile_w_arnings
${ bld} /makefile_database_files: ${ b ld} /makefile_database ${ b ld} /check_makefile_w_arnings
@sed -n -e '/^# Files$$/,/^# files hash-table stats:$$/p' $< > $@
build/makefile_built_directories : build /check_makefile_w_arnings
${ bld} /makefile_built_directories: ${ b ld} /check_makefile_w_arnings
@echo ${ more_built_directories } | tr ' ' '\n' | grep -v '^[[:space:]]*$$' | sort > $@
bui ld/makefile_built_files: b ui ld/check_makefile_w_arnings
${ bld} /makefile_built_files: ${ b ld} /check_makefile_w_arnings
@echo ${ built_files } | tr ' ' '\n' | grep -v '^[[:space:]]*$$' | sort > $@
bui ld/makefile_phony: b ui ld/makefile_database_files b ui ld/check_makefile_w_arnings
${ bld} /makefile_phony: ${ b ld} /makefile_database_files ${ b ld} /check_makefile_w_arnings
@sed -n -e 's/^\.PHONY: \(.*\)$$/\1/p' $< | tr ' ' '\n' | grep -v '^[[:space:]]*$$' | sort > $@
bui ld/makefile_targets: b ui ld/makefile_database_files b ui ld/check_makefile_w_arnings
${ bld} /makefile_targets: ${ b ld} /makefile_database_files ${ b ld} /check_makefile_w_arnings
@grep -E -v '^([[:space:]]|#|\.|$$|^[^:]*:$$)' $< | grep '^[^ :]*:' | sed -e 's|^\([^:]*\):.*$$|\1|' | sort > $@
bui ld/makefile_non_file_targets: b ui ld/makefile_phony b ui ld/makefile_built_directories b ui ld/check_makefile_w_arnings
@cat bui ld/makefile_phony bui ld/makefile_built_directories | sort > $@
${ bld} /makefile_non_file_targets: ${ b ld} /makefile_phony ${ b ld} /makefile_built_directories ${ b ld} /check_makefile_w_arnings
@cat ${ bld} /makefile_phony ${ bld} /makefile_built_directories | sort > $@
bui ld/makefile_file_targets: b ui ld/makefile_non_file_targets b ui ld/makefile_targets b ui ld/check_makefile_w_arnings
@comm -23 bui ld/makefile_targets bui ld/makefile_non_file_targets > $@
${ bld} /makefile_file_targets: ${ b ld} /makefile_non_file_targets ${ b ld} /makefile_targets ${ b ld} /check_makefile_w_arnings
@comm -23 ${ bld} /makefile_targets ${ bld} /makefile_non_file_targets > $@
${built_directories} : b ui ld/check_makefile
${built_directories} : ${ b ld} /check_makefile
${more_built_directories} : Makefile
mkdir -p $@ && touch $@
# 32k header of the ISO9660 image
bui ld/os.32k: example -os /os .asm b ui ld/check_makefile
${ bld} /os.32k: example -os /os .asm ${ b ld} /check_makefile
nasm -w+macro-params -w+macro-selfref -w+orphan-labels -w+gnu-elf-extensions -o $@ $<
build/os.iso : build /iso_files /os .zip build /iso_files /boot /iso_boot .sys build /check_makefile
mkisofs \
# Circumvent the fact that faketime does not work on system binaries in macos
./utils/mkisofs ./utils/mformat ./utils/mcopy : ${bld }/check_makefile # TODO: depend on the mkisofs binary
cp $$ ( which $$ ( basename $@ ) ) $@
chmod u+x $@
cp_T_option = $$ ( if test " $$ (uname -s) " = Darwin; then echo '' ; else echo '-T' ; fi )
${bld}/os.iso : ${bld }/iso_files /os .zip ${bld }/iso_files /boot /iso_boot .sys ./utils /mkisofs ${bld }/check_makefile
! test -d ${ bld } /iso_files.tmp
cp -a ${ cp_T_option } -- ${ bld } /iso_files ${ bld } /iso_files.tmp
find ${ bld } /iso_files.tmp -depth -exec touch -t ${ commit_timestamp } '{}' ';'
UTILS = " $$ PWD/utils " ./utils/faketime.sh ${ commit_timestamp_iso_8601 } sh -c ' ( cd ./${ bld } /iso_files.tmp/ && " $$ UTILS/mkisofs " \
--input-charset utf-8 \
-rock \
-joliet \
@ -159,18 +231,22 @@ build/os.iso: build/iso_files/os.zip build/iso_files/boot/iso_boot.sys build/che
-no-emul-boot \
-boot-load-size 4 \
-pad \
-output $@ \
./build/iso_files/
-output ../os.iso \
.) '
rm -- ${ bld } /iso_files.tmp/os.zip \
${ bld } /iso_files.tmp/boot/iso_boot.sys
rmdir ${ bld } /iso_files.tmp/boot/
rmdir ${ bld } /iso_files.tmp/
# Layout:
# MBR; GPT; UNIX sh & MS-DOS batch scripts; ISO9660; FAT12; GPT mirror; ZIP
d e f i n e o f f s e t
tmp_${1} = ${ 3 }
bui ld/offsets/${1}.dec: $${tmp_ ${1}:%=b ui ld/offsets /%.dec } ${4} b ui ld/check_makefile
${ bld} /offsets/${1}.dec: $${tmp_ ${1}:%=${ b ld} /offsets /%.dec } ${4} ${ b ld} /check_makefile
echo $$ $$ ( ( ${ 2 } ) ) | tee $$ @
${1} = $$ $$ ( cat bui ld/offsets/${ 1 } .dec)
dep_${1} = bui ld/offsets/${ 1 } .dec
${1} = $$ $$ ( cat ${ bld} /offsets/${ 1 } .dec)
dep_${1} = ${ bld} /offsets/${ 1 } .dec
e n d e f
d e f i n e d i v _ r o u n d _ u p
@ -184,12 +260,12 @@ ${eval ${call offset,sectors_os_size, $${bytes_os_size} / $${sector_size},
$ { e v a l $ { c a l l o f f s e t , t r a c k s _ o s _ s i z e , $$ { s e c t o r s _ o s _ s i z e } / $$ { o s _ f l o p p y _ c h s _ s } , s e c t o r s _ o s _ s i z e , } }
# round up
$ { e v a l $ { c a l l o f f s e t , b y t e s _ i s o _ s i z e , $$ $$ ( u t i l s / f i l e - l e n g t h . s h - c bu i l d / o s . i s o ) , , b u i l d / o s . i s o } }
$ { e v a l $ { c a l l o f f s e t , b y t e s _ i s o _ s i z e , $$ $$ ( u t i l s / f i l e - l e n g t h . s h - c ${ b l d } / o s . i s o ) , , $ { b l d } / o s . i s o } }
$ { e v a l $ { c a l l o f f s e t , s e c t o r s _ i s o _ s i z e , $ { c a l l d i v _ r o u n d _ u p , $$ { b y t e s _ i s o _ s i z e } , $$ { s e c t o r _ s i z e } } , b y t e s _ i s o _ s i z e , } }
$ { e v a l $ { c a l l o f f s e t , t r a c k s _ i s o _ s i z e , $ { c a l l d i v _ r o u n d _ u p , $$ { s e c t o r s _ i s o _ s i z e } , $$ { o s _ f l o p p y _ c h s _ s } } , s e c t o r s _ i s o _ s i z e , } }
# round up
$ { e v a l $ { c a l l o f f s e t , b y t e s _ z i p _ s i z e , $$ $$ ( u t i l s / f i l e - l e n g t h . s h - c bu i l d / o s . z i p ) , , b u i l d / o s . z i p } }
$ { e v a l $ { c a l l o f f s e t , b y t e s _ z i p _ s i z e , $$ $$ ( u t i l s / f i l e - l e n g t h . s h - c ${ b l d } / o s . z i p ) , , $ { b l d } / o s . z i p } }
$ { e v a l $ { c a l l o f f s e t , s e c t o r s _ z i p _ s i z e , $ { c a l l d i v _ r o u n d _ u p , $$ { b y t e s _ z i p _ s i z e } , $$ { s e c t o r _ s i z e } } , b y t e s _ z i p _ s i z e , } }
$ { e v a l $ { c a l l o f f s e t , t r a c k s _ z i p _ s i z e , $ { c a l l d i v _ r o u n d _ u p , $$ { s e c t o r s _ z i p _ s i z e } , $$ { o s _ f l o p p y _ c h s _ s } } , s e c t o r s _ z i p _ s i z e , } }
@ -222,16 +298,17 @@ ${eval ${call offset,bytes_gpt_mirror_start, $${bytes_gpt_mirror_end} - $${bytes
$ { e v a l $ { c a l l o f f s e t , b y t e s _ z i p _ e n d , $$ { b y t e s _ o s _ s i z e } , , } }
os_fat12_partition = " $@ @@ ${ bytes_fat12_start } "
build/os.fat12 : build /os .zip ${dep_bytes_fat 12_size } ${dep_bytes_fat 12_start } ${dep_sectors_os_size } build /check_makefile
${bld}/os.fat12 : ${bld }/os .zip ${dep_bytes_fat 12_size } ${dep_bytes_fat 12_start } ${dep_sectors_os_size } \
./utils/mformat ./utils/mcopy ${ bld } /check_makefile
set -x; dd if = /dev/zero bs = ${ sector_size } count = ${ sectors_os_size } of = $@
set -x; mformat -v "Example OS" \
./utils/faketime.sh ${ commit_timestamp_iso_8601 } ./utils/ mformat -v "Example OS" \
-T ${ sectors_fat12_size } \
-h ${ os_floppy_chs_h } \
-s ${ os_floppy_chs_s } \
-i ${ os_fat12_partition }
set -x; mcopy -i ${ os_fat12_partition } build/os.zip "::os.zip"
./utils/faketime.sh ${ commit_timestamp_iso_8601 } ./utils/mcopy -i ${ os_fat12_partition } $< "::os.zip"
bui ld/iso_files/os.zip: b ui ld/os .zip b ui ld/check_makefile
${ bld} /iso_files/os.zip: ${ b ld} /os .zip ${ b ld} /check_makefile
# TODO: make it so that the various file formats are mutual quines:
# * the ISO should contain the original file
# * the ZIP should contain the original file
@ -239,21 +316,35 @@ build/iso_files/os.zip: build/os.zip build/check_makefile
cp $< $@
# 4 sectors loaded when booting from optical media (CD-ROM, …):
bui ld/iso_files/boot/iso_boot.sys: b ui ld/os .32k b ui ld/check_makefile
${ bld} /iso_files/boot/iso_boot.sys: ${ b ld} /os .32k ${ b ld} /check_makefile
# TODO: this copy of the (or alternate) bootsector should contain a Boot Information Table,
# see https://wiki.osdev.org/El-Torito#A_BareBones_Boot_Image_with_Boot_Information_Table
dd if = $< bs = 512 count = 4 of = $@
build/os.zip : build /os .32k build /check_makefile
zip $@ $<
${bld}/os.zip : ${bld }/os .32k ${bld }/check_makefile
# We copy os.32k and alter its timestamp to ensure reproducible
# builds.
mkdir -p ${ bld } /os.32k.tmp
cp -a $< ${ bld } /os.32k.tmp/os.32k
touch -t ${ commit_timestamp } ${ bld } /os.32k.tmp/os.32k
( cd ${ bld } /os.32k.tmp/ && zip -X ../os.zip os.32k)
rm ${ bld } /os.32k.tmp/os.32k
rmdir ${ bld } /os.32k.tmp
build/os.zip.adjusted : build /os .zip ${dep_bytes_zip_start } build /check_makefile
${ bld} /os.zip.adjusted: ${ b ld} /os .zip ${dep_bytes_zip_start } ${ b ld} /check_makefile
# TODO: the ZIP file can end with a variable-length comment, this would allow us to hide the GPT mirrors.
set -x; dd if = /dev/zero bs = 1 count = ${ bytes_zip_start } of = $@
cat $< >> $@
zip --adjust-sfx $@
${os_filename} : build /os .32k build /os .iso build /os .fat 12 build /os .zip .adjusted \
gdisk_pipe_commands_slowly = while read str; do echo " $$ str " ; printf "\033[1;33m%s\033[m\n" " $$ str " >& 2; sleep 0.01; done
commit_hash_as_guid = $$ ( git log -1 --pretty= format:%H | sed -e 's/^\(.\{8\}\)\(.\{4\}\)\(.\{4\}\)\(.\{4\}\)\(.\{12\}\).*$$/\1-\2-\3-\4-\5/' | tr '[:lower:]' '[:upper:]' )
git_dirty = test -n " $$ (git diff --shortstat) "
gpt_disk_guid = ${ commit_hash_as_guid } $$ ( if $$ git_dirty; then printf '0' ; else printf '2' ; fi )
gpt_partition_guid = ${ commit_hash_as_guid } $$ ( if $$ git_dirty; then printf '1' ; else printf '3' ; fi )
${os_filename} : ${bld }/os .32k ${bld }/os .iso ${bld }/os .fat 12 ${bld }/os .zip .adjusted \
${ dep_bytes_header_32k_start } \
${ dep_bytes_header_32k_size } \
${ dep_bytes_fat12_start } \
@ -263,14 +354,14 @@ ${os_filename}: build/os.32k build/os.iso build/os.fat12 build/os.zip.adjusted \
${ dep_sectors_fat12_start } \
${ dep_sectors_fat12_size } \
${ dep_bytes_zip_start } \
bui ld/check_makefile
${ bld} /check_makefile
rm -f $@
# start with the .iso
cp bui ld/os.iso $@
cp ${ bld} /os.iso $@
# splice in the first 32k (bootsector and partition table)
set -x; dd skip = ${ bytes_header_32k_start } seek = ${ bytes_header_32k_start } bs = 1 count = ${ bytes_header_32k_size } conv = notrunc if = bui ld/os.32k of = $@
set -x; dd skip = ${ bytes_header_32k_start } seek = ${ bytes_header_32k_start } bs = 1 count = ${ bytes_header_32k_size } conv = notrunc if = ${ bld} /os.32k of = $@
# splice in fat12
set -x; dd skip = ${ bytes_fat12_start } seek = ${ bytes_fat12_start } bs = 1 count = ${ bytes_fat12_size } conv = notrunc if = bui ld/os.fat12 of = $@
set -x; dd skip = ${ bytes_fat12_start } seek = ${ bytes_fat12_start } bs = 1 count = ${ bytes_fat12_size } conv = notrunc if = ${ bld} /os.fat12 of = $@
# pad with zeroes to prepare for GPT table
set -x; dd if = /dev/zero seek = $$ ( ( ${ bytes_gpt_mirror_end } - 1 ) ) bs = 1 count = 1 conv = notrunc of = $@
# patch the partition table
@ -281,13 +372,21 @@ ${os_filename}: build/os.32k build/os.iso build/os.fat12 build/os.zip.adjusted \
# * Recovery and transformation options, make Hybrid,
# * add GPT partition #1 to the hybrid MBR, do Not put the EFI partition first,
# * MBR partition type=0x01, bootable=Yes, do Not add extra partitions,
# * Print GPT, print MBR, Write, Proceed.
# * back to Main menu,
# * eXpert mode,
# * Change partition GUID, the GUID itself,
# * change disk GUID, the GUID itself,
# * Print GPT, print prOtective MBR, Write, Proceed.
( if test " $$ (uname -o) " = "Cygwin" ; then printf "Y\n" ; fi ; \
printf " d\nx\nl\n1\nm\nn\n1\n ${ sectors_fat12_start } \n ${ sectors_fat12_size } \n0700\n " ; \
printf "d\nx\nl\n1\nm\n" ; \
printf " n\n1\n ${ sectors_fat12_start } \n ${ sectors_fat12_size } \n0700\n " ; \
printf "r\nh\n" ; \
printf "1\nN\n" ; \
printf "01\nY\nN\n" ; \
printf "p\no\nw\nY\n" ) | while read str; do echo " $$ str " ; printf "\033[1;33m%s\033[m\n" " $$ str " >& 2; sleep 0.01; done | gdisk $@
printf "x\n" ; \
printf " g\n ${ gpt_disk_guid } \n " ; \
printf " c\n ${ gpt_partition_guid } \n " ; \
printf "p\no\nw\nY\n" ) | ${ gdisk_pipe_commands_slowly } | gdisk $@
# Inject MS-DOS newlines (CR+LF) and comments (":: ") in the GUID field of unused partition table entries,
# so that the part that is to be skipped by MS-DOS does not form a line longer than the MS-DOS maximum
# line length (8192 excluding CR+LF). $i below is the partition entry number, starting from 1
@ -297,123 +396,156 @@ ${os_filename}: build/os.32k build/os.iso build/os.fat12 build/os.zip.adjusted \
printf "\r\n:: %02x" $$ i | dd bs = 1 seek = $$ ( ( ${ bytes_gpt_mirror_start } + ( ( $$ i) - 1) * 128 + 16) ) count = 7 conv = notrunc of = $@ ; \
done
# splice in zip at the end
set -x; dd skip = ${ bytes_zip_start } seek = ${ bytes_zip_start } bs = 1 conv = notrunc if = bui ld/os.zip.adjusted of = $@
set -x; dd skip = ${ bytes_zip_start } seek = ${ bytes_zip_start } bs = 1 conv = notrunc if = ${ bld} /os.zip.adjusted of = $@
chmod a+x-w $@
bui ld/os.file: ${os_filename } b ui ld/check_makefile
${ bld} /os.file: ${os_filename } ${ b ld} /check_makefile
file -kr $< > $@
bui ld/os.gdisk: ${os_filename } b ui ld/check_makefile
${ bld} /os.gdisk: ${os_filename } ${ b ld} /check_makefile
# gdisk commands:
# * Print partition table
# * Recovery and transformation options
# * print prOtective MBR table
# * Quit
printf 'p\nr\no\nq\n' | gdisk $< | tee $@
# * eXpert mode
# * Print partition table
# * print detailed Information about the (only) partition
# * print prOtective MBR table
# * Quit
printf '2\nx\np\ni\nr\no\nq\n' | ${ gdisk_pipe_commands_slowly } | gdisk $< | tee $@
bui ld/os.offsets.hex: ${offset_names :%=b ui ld/offsets /%.hex } b ui ld/check_makefile
grep '^' ${ offset_names : %= bui ld/offsets/%.hex} | sed -e 's/:/: 0x/' | column -t > $@
${ bld} /os.offsets.hex: ${offset_names :%=${ b ld} /offsets /%.hex } ${ b ld} /check_makefile
grep '^' ${ offset_names : %= ${ bld} /offsets/%.hex} | sed -e 's|^.*/||' -e 's/:/: 0x/' | column -t > $@
bui ld/os.offsets.dec: ${offset_names :%=b ui ld/offsets /%.dec } b ui ld/check_makefile
grep '^' ${ offset_names : %= bui ld/offsets/%.dec} | sed -e 's/:/: /' | column -t > $@
${ bld} /os.offsets.dec: ${offset_names :%=${ b ld} /offsets /%.dec } ${ b ld} /check_makefile
grep '^' ${ offset_names : %= ${ bld} /offsets/%.dec} | sed -e 's|^.*/||' -e 's/:/: /' | column -t > $@
bui ld/offsets/%.hex: b ui ld/offsets /%.dec
${ bld} /offsets/%.hex: ${ b ld} /offsets /%.dec
printf '%x\n' $$ ( cat $<) > $@
bui ld/os.hex_with_offsets: ${os_filename } b ui ld/os .offsets .hex
${ bld} /os.hex_with_offsets: ${os_filename } ${ b ld} /os .offsets .hex
hexdump -C $< \
| grep -E -e " ( $$ (cat bui ld/os.offsets.hex | cut -d '=' -f 2 | sed -e 's/^[[:space:]]*0x\(.*\).$$ /^\10/' | tr '\n' '|')^) " --color= yes > $@
| grep -E -e " ( $$ (cat ${ bld} /os.offsets.hex | cut -d '=' -f 2 | sed -e 's/^[[:space:]]*0x\(.*\).$$ /^\10/' | tr '\n' '|')^) " --color= yes > $@
bui ld/os.ndisasm.disasm: ${os_filename } utils /compact -ndisasm .sh b ui ld/check_makefile
${ bld} /os.ndisasm.disasm: ${os_filename } utils /compact -ndisasm .sh ${ b ld} /check_makefile
./utils/compact-ndisasm.sh $< $@
bui ld/os.reasm.asm: b ui ld/os .ndisasm .disasm b ui ld/check_makefile
${ bld} /os.reasm.asm: ${ b ld} /os .ndisasm .disasm ${ b ld} /check_makefile
sed -e 's/^[^ ]\+ \+[^ ]\+ \+//' $< > $@
bui ld/test_pass/noemu_%.reasm bui ld/%.reasm: b ui ld/%.reasm .asm ${os_filename } utils /compact -ndisasm .sh b ui ld/check_makefile
${ bld} /test_pass/noemu_%.reasm ${ bld} /%.reasm: ${ b ld} /%.reasm .asm ${os_filename } utils /compact -ndisasm .sh ${ b ld} /check_makefile
# For now ignore this test, since we cannot have a reliable re-assembly of arbitrary data.
touch bui ld/test_pass/noemu_$* .reasm bui ld/$* .reasm
touch ${ bld} /test_pass/noemu_$* .reasm ${ bld} /$* .reasm
# nasm $< -o $@
# @echo "diff $@ ${os_filename}"
# @diff $@ ${os_filename} \
# && echo " [1;32mRe-assembled file is identical to ${os_filename} [m" \
# || (./utils/compact-ndisasm.sh $@ bui ld/os.reasm.disasm; \
# echo " [0;33mRe-assembled file is different from ${os_filename}. Use meld bui ld/os.ndisasm.disasm bui ld/os.reasm.disasm to see differences. [m"; \
# || (./utils/compact-ndisasm.sh $@ ${ bld} /os.reasm.disasm; \
# echo " [0;33mRe-assembled file is different from ${os_filename}. Use meld ${ bld} /os.ndisasm.disasm ${ bld} /os.reasm.disasm to see differences. [m"; \
# exit 0)
#os.arm.disasm: ${os_filename} bui ld/check_makefile
#os.arm.disasm: ${os_filename} ${ bld} /check_makefile
# arm-none-eabi-objdump --endian=little -marm -b binary -D --adjust-vma=0x8000 $< > $@
.PHONY : clean
clean : build /check_makefile
rm -f ${ built_files }
for d in $$ ( echo ${ more_built_directories } | tr ' ' '\n' | sort --reverse) ; do \
clean : ${bld }/check_makefile
rm -f ${ built_files } ${ temp_files } ${ bld } /${ reproducible_os_filename }
if test -d ${ bld } /reproducible; then \
unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make OS_FILENAME = ${ reproducible_os_filename } \
BUILD_DIR = ${ bld } /reproducible \
SCREENSHOTS_DIR = ${ bld } /reproducible/screenshots \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
clean; \
fi
for d in $$ ( echo ${ more_built_directories } ${ temp_directories } | tr ' ' '\n' | sort --reverse) ; do \
if test -e " $$ d " ; then \
rmdir " $$ d " ; \
fi ; \
done
.gitignore : build /check_makefile
.gitignore : ${ b ld} /check_makefile
for f in ${ built_files } ; do echo " / $$ f " ; done | sort > $@
.PHONY : test
test : ${tests_emu :test /%=b ui ld/test_pass /emu_ %} \
${ tests_noemu : test /%= bui ld/test_pass/noemu_%} \
${ tests_requiring_sudo : test /%= bui ld/test_pass/sudo_%} \
test : ${tests_emu :test /%=${ b ld} /test_pass /emu_ %} \
${ tests_noemu : test /%= ${ bld} /test_pass/noemu_%} \
${ tests_requiring_sudo : test /%= ${ bld} /test_pass/sudo_%} \
all \
bui ld/check_makefile
${ bld} /check_makefile
.PHONY : ${tests_emu }
${tests_emu} : build /test_pass /emu_ $$( @F )
.PHONY : test /emu test /noemu test /requiring_sudo
test/emu : ${tests_emu } ${bld }/check_makefile
test/requiring_sudo : ${tests_requiring_sudo } ${bld }/check_makefile
test/noemu : ${tests_noemu } ${bld }/check_makefile
build/test_pass/emu_% deploy-screenshots/%.png deploy-screenshots/%-anim.gif : \
.PHONY : ${tests_emu } ${tests_noemu } ${tests_requiring_sudo }
${tests_emu} : ${bld }/test_pass /emu_ $$( @F ) ${bld }/check_makefile
${tests_noemu} : ${bld }/test_pass /noemu_ $$( @F ) ${bld }/check_makefile
${tests_requiring_sudo} : ${bld }/test_pass /sudo_ $$( @F ) ${bld }/check_makefile
${bld}/test_pass/emu_% ${screenshots}/%.png ${screenshots}/%-anim.gif : \
${ os_filename } \
build/checkerboard_800x600.xbm \
${ bld} /checkerboard_800x600.xbm \
utils/gui-wrapper.sh utils/ansi-screenshots/ansi_screenshot.sh utils/ansi-screenshots/to_ansi.sh \
test/%.sh \
build/check_makefile \
| build/test_pass deploy-screenshots
${ bld} /check_makefile \
| ${ bld } /test_pass ${ screenshots }
./utils/gui-wrapper.sh 800x600x24 ./test/$* .sh $<
touch bui ld/test_pass/emu_$*
touch ${ bld} /test_pass/emu_$*
.PHONY : test /noemu
test/noemu : ${tests_noemu :test /%=build /test_pass /noemu_ %} build /check_makefile
build/test_pass/noemu_zip : ${os_filename } build /check_makefile
${bld}/test_pass/noemu_zip : ${os_filename } ${bld }/check_makefile
unzip -t ${ os_filename }
touch $@
bui ld/test_pass/noemu_sizes: b ui ld/os .32k ${os_filename } b ui ld/check_makefile
test " $$ (utils/file-length.sh -c bui ld/os.32k)" = " $$ ((32*1024)) "
${ bld} /test_pass/noemu_sizes: ${ b ld} /os .32k ${os_filename } ${ b ld} /check_makefile
test " $$ (utils/file-length.sh -c ${ bld} /os.32k)" = " $$ ((32*1024)) "
test " $$ (utils/file-length.sh -c ${ os_filename } ) " = " $$ ((1440*1024)) "
touch $@
# check that the fat filesystem has the correct contents
bui ld/test_pass/noemu_fat12_contents: ${os_filename } ${dep_bytes_fat 12_start } b ui ld/check_makefile
${ bld} /test_pass/noemu_fat12_contents: ${os_filename } ${dep_bytes_fat 12_start } ${ b ld} /check_makefile
mdir -i " $<@@ ${ bytes_fat12_start } " :: | grep -E "^os[[:space:]]+zip[[:space:]]+"
touch $@
.PHONY : test /requiring_sudo
test/requiring_sudo : ${tests_requiring_sudo :test /%=b ui ld/test_pass /sudo_ %} b ui ld/check_makefile
test/requiring_sudo : ${tests_requiring_sudo :test /%=${ b ld} /test_pass /sudo_ %} ${ b ld} /check_makefile
# check that the fat filesystem can be mounted and has the correct contents
build/test_pass/sudo_fat12_mount : ${os_filename } ${dep_bytes_fat 12_start } build /check_makefile | build /mnt_fat 12
sudo umount build/mnt_fat12 || true
sudo mount -o loop,ro,offset= ${ bytes_fat12_start } $< build/mnt_fat12
ls -l build/mnt_fat12 | grep os.zip
sudo umount build/mnt_fat12
${bld}/test_pass/sudo_fat12_mount : ${os_filename } ${dep_bytes_fat 12_start } ${bld }/check_makefile | ${bld }/mnt_fat 12
sudo umount ${ bld } /mnt_fat12 || true
# debug failure to mount the FAT12 filesystem
( set -x ; \
grep '^' ${ bld } /offsets/* ; \
( sudo mount -o loop,ro,offset= ${ bytes_fat12_start } $< ${ bld } /mnt_fat12) || true ; \
dmesg | tail ; \
ls -l ${ bld } /mnt_fat12 | ( grep os.zip || true ) ; \
sudo umount ${ bld } /mnt_fat12 || true ; \
sleep 10 ; \
cat ${ os_filename } | xz | base64 ; \
( sudo mount -o loop,ro,offset= ${ bytes_fat12_start } $< ${ bld } /mnt_fat12) || true ; \
sudo umount ${ bld } /mnt_fat12 || true ; \
) > debug-fat12.log 2>& 1
cat debug-fat12.log
sudo mount -o loop,ro,offset= ${ bytes_fat12_start } $< ${ bld } /mnt_fat12
ls -l ${ bld } /mnt_fat12 | grep os.zip
sudo umount ${ bld } /mnt_fat12
touch $@
build/test_pass/sudo_iso_mount : ${os_filename } build /check_makefile | build /mnt_iso
sudo umount build/mnt_iso || true
grep '^' build/offsets/* # debug failure to mount the ISO9660 filesystem
( sudo mount -o loop,ro $< build/mnt_iso) || true
dmesg | tail # debug failure to mount the ISO9660 filesystem
hexdump -C os.bat
ls -l build/mnt_iso | grep os.zip
sudo umount build/mnt_iso
sudo mount -o loop,ro $< build/mnt_iso
sudo umount build/mnt_iso
${bld}/test_pass/sudo_iso_mount : ${os_filename } ${bld }/check_makefile | ${bld }/mnt_iso
sudo umount ${ bld } /mnt_iso || true
# debug failure to mount the ISO9660 filesystem
( set -x ; \
grep '^' ${ bld } /offsets/* ; \
( sudo mount -o loop,ro $< ${ bld } /mnt_iso) || true ; \
dmesg | tail ; \
cat ${ os_filename } | xz | base64 ; \
ls -l ${ bld } /mnt_iso | grep os.zip ; \
sudo umount ${ bld } /mnt_iso || true ; \
sleep 11 ; \
( sudo mount -o loop,ro $< ${ bld } /mnt_iso) || true ; \
sudo umount ${ bld } /mnt_iso || true ; \
) > debug-iso.log 2>& 1
cat debug-iso.log
sudo mount -o loop,ro $< ${ bld } /mnt_iso
sudo umount ${ bld } /mnt_iso
touch $@
.PHONY : test /macos
@ -425,38 +557,80 @@ test/macos-sh-x11:
sudo chmod a+rwxt /tmp/.X11-unix
xvfb :42 & \
sleep 5; \
DISPLAY = :42 xterm -e ./os.bat & \
DISPLAY = :42 xterm -e ./${ os_filename } & \
sleep 5; \
# DISPLAY=:42 import -window root deploy-screenshots /macos-sh-x11.png
screencapture deploy-screenshots /macos-sh-x11-screencapture.png
# DISPLAY=:42 import -window root ${screenshots} /macos-sh-x11.png
screencapture ${ screenshots } /macos-sh-x11-screencapture.png
.PHONY : test /macos -sh
test/macos-sh : b ui ld/check_makefile \
bui ld/checkerboard_1024x768.png \
| deploy-screenshots
osascript -e 'tell app "Terminal" to do script "' " $$ PWD " '/ os.bat "'
test/macos-sh : ${ b ld} /check_makefile \
${ bld} /checkerboard_1024x768.png \
| ${ screenshots }
osascript -e 'tell app "Terminal" to do script "' " $$ PWD " '/ ${os_filename} "'
sleep 2
osascript -e 'tell app "Terminal" to activate'
sleep 5
( date +%n && sleep 0.2 && date +%n) || true
screencapture deploy-screenshots /screencapture-os-bat.png
screencapture ${ screenshots } /screencapture-os-bat.png
./utils/gui-wrapper-mac.sh 1024x768x24 ./test/gui-sh-mac.sh ${ os_filename }
# See https://wiki.osdev.org/EFI#Emulation to emulate an UEFI system with qemu, to test the EFI boot from hdd / cd / fd (?).
# Create checkerboard background
bui ld/checkerboard_%.png: b ui ld/check_makefile
${ bld} /checkerboard_%.png: ${ b ld} /check_makefile
convert -size " $* " \
tile:pattern:checkerboard \
-auto-level +level-colors 'gray(192),gray(128)' \
$@
bui ld/checkerboard_%.xbm: b ui ld/check_makefile
${ bld} /checkerboard_%.xbm: ${ b ld} /check_makefile
convert -size " $* " \
tile:pattern:checkerboard \
-auto-level \
$@
# Temporary files
build/bochsrc build/bochscontinue build/twm_cfg build/virtualbox.img :
${bld}/bochsrc ${bld}/bochscontinue ${bld}/twm_cfg ${bld}/virtualbox.img : ${bld }/check_makefile
touch $@
# DEBUG: ${bld}/os.hex_with_offsets ${bld}/os.offsets.hex
${bld}/test_pass/noemu_reproducible_build : ${os_filename } ${bld }/os .hex_with_offsets ${bld }/check_makefile
# Let some time pass so that any timestamp that may affect the result changes.
sleep 5
# TODO: try to see if we can re-enable some of these variables without
# causing problems on macos.
unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make OS_FILENAME = ${ reproducible_os_filename } \
BUILD_DIR = ${ bld } /reproducible \
SCREENSHOTS_DIR = ${ bld } /reproducible/screenshots \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
clean
unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make OS_FILENAME = ${ reproducible_os_filename } \
BUILD_DIR = ${ bld } /reproducible \
SCREENSHOTS_DIR = ${ bld } /reproducible/screenshots \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
${ reproducible_os_filename } \
${ bld } /reproducible/os.hex_with_offsets
# Check that the second build produced the same file.
if ! diff ${ os_filename } ${ reproducible_os_filename } ; then \
diff ${ bld } /os.hex_with_offsets ${ bld } /reproducible/os.hex_with_offsets || true; \
if test " $$ (uname -s) " = Darwin -o " $$ (uname -o) " = "Cygwin" ; then \
for i in ` seq 5` ; do \
printf '\033[1;31m########################################################\033[m' ; \
done ; \
echo "REPRODUCIBLE BUILDS ARE UNSUPPORTED ON MACOS AND WINDOWS" ; \
for i in ` seq 5` ; do \
printf '\033[1;31m########################################################\033[m' ; \
done ; \
else \
exit 1; \
fi ; \
fi
unset MAKEFLAGS MAKELEVEL MAKE_TERMERR MFLAGS; \
make OS_FILENAME = ${ reproducible_os_filename } \
BUILD_DIR = ${ bld } /reproducible \
SCREENSHOTS_DIR = ${ bld } /reproducible/screenshots \
COMMIT_TIMESTAMP_ISO_8601 = ${ COMMIT_TIMESTAMP_ISO_8601 } \
clean
touch $@