# Nuke built-in rules. .SUFFIXES: override OUTPUT := os # User controllable toolchain and toolchain prefix. TOOLCHAIN := TOOLCHAIN_PREFIX := ifneq ($(TOOLCHAIN),) ifeq ($(TOOLCHAIN_PREFIX),) TOOLCHAIN_PREFIX := $(TOOLCHAIN)- endif endif # User controllable C compiler command. ifneq ($(TOOLCHAIN_PREFIX),) CC := $(TOOLCHAIN_PREFIX)gcc else CC := cc endif # User controllable linker command. LD := $(TOOLCHAIN_PREFIX)ld # Defaults overrides for variables if using "llvm" as toolchain. ifeq ($(TOOLCHAIN),llvm) CC := clang LD := ld.lld endif # User controllable C flags. CFLAGS := -g -O2 -pipe -Iinclude # User controllable C preprocessor flags. We set none by default. CPPFLAGS := # User controllable nasm flags. NASMFLAGS := -g # User controllable linker flags. We set none by default. LDFLAGS := # Check if CC is Clang. override CC_IS_CLANG := $(shell ! $(CC) --version 2>/dev/null | grep -q '^Target: '; echo $$?) # If the C compiler is Clang, set the target as needed. ifeq ($(CC_IS_CLANG),1) override CC += \ -target x86_64-unknown-none-elf endif # Internal C flags that should not be changed by the user. override CFLAGS += \ -Wall \ -Wextra \ -std=gnu11 \ -ffreestanding \ -fno-stack-protector \ -fno-stack-check \ -fno-lto \ -fno-PIC \ -ffunction-sections \ -fdata-sections \ -m64 \ -march=x86-64 \ -mabi=sysv \ -mno-80387 \ -mno-mmx \ -mno-sse \ -mno-sse2 \ -mno-red-zone \ -mcmodel=kernel # Internal C preprocessor flags that should not be changed by the user. override CPPFLAGS := \ -I src \ $(CPPFLAGS) \ -MMD \ -MP # Internal nasm flags that should not be changed by the user. override NASMFLAGS := \ -f elf64 \ $(patsubst -g,-g -F dwarf,$(NASMFLAGS)) \ -Wall # Internal linker flags that should not be changed by the user. override LDFLAGS += \ -m elf_x86_64 \ -nostdlib \ -static \ -z max-page-size=0x1000 \ --gc-sections \ -T linker.ld # Use "find" to glob all *.c, *.S, and *.asm files in the tree and obtain the # object and header dependency file names. override SRCFILES := $(shell find -L src -type f 2>/dev/null | LC_ALL=C sort) override CFILES := $(filter %.c,$(SRCFILES)) override ASFILES := $(filter %.S,$(SRCFILES)) override NASMFILES := $(filter %.asm,$(SRCFILES)) override OBJ := $(addprefix obj/,$(CFILES:.c=.c.o) $(ASFILES:.S=.S.o) $(NASMFILES:.asm=.asm.o)) override HEADER_DEPS := $(addprefix obj/,$(CFILES:.c=.c.d) $(ASFILES:.S=.S.d)) # Default target. This must come first, before header dependencies. .PHONY: all all: bin/$(OUTPUT) sh limine.sh sh run.sh # Include header dependencies. -include $(HEADER_DEPS) # Link rules for the final executable. bin/$(OUTPUT): linker.ld $(OBJ) mkdir -p "$(dir $@)" $(LD) $(LDFLAGS) $(OBJ) -o $@ # Compilation rules for *.c files. obj/%.c.o: %.c Makefile mkdir -p "$(dir $@)" $(CC) $(CFLAGS) $(CPPFLAGS) -c $< -o $@ # Compilation rules for *.S files. obj/%.S.o: %.S Makefile mkdir -p "$(dir $@)" $(CC) $(CFLAGS) $(CPPFLAGS) -c $< -o $@ # Compilation rules for *.asm (nasm) files. obj/%.asm.o: %.asm Makefile mkdir -p "$(dir $@)" nasm $(NASMFLAGS) $< -o $@ usb: all sudo dd if=image.hdd of=/dev/sda bs=4M oflag=sync status=progress # Remove object files and the final executable. .PHONY: clean clean: rm -rf bin obj