mkfile_path := $(abspath $(lastword $(MAKEFILE_LIST))) mkfile_dir := $(dir $(mkfile_path)) # Make a variable lazily evaluated at first call # From https://blog.jgc.org/2016/07/lazy-gnu-make-variables.html make-lazy = $(eval $1 = $$(eval $1 := $(value $(1)))$$($1)) export KALEIDOSCOPE_DIR ?= $(abspath $(mkfile_dir)/..) export KALEIDOSCOPE_BIN_DIR ?= $(KALEIDOSCOPE_DIR)/bin # Arduino CLI config export ARDUINO_CONTENT ?= $(KALEIDOSCOPE_DIR)/.arduino export ARDUINO_DIRECTORIES_DATA ?= $(ARDUINO_CONTENT)/data export ARDUINO_DIRECTORIES_DOWNLOADS ?= $(ARDUINO_CONTENT)/downloads export ARDUINO_DIRECTORIES_USER ?= $(ARDUINO_CONTENT)/user export ARDUINO_CLI_CONFIG ?= $(ARDUINO_DIRECTORIES_DATA)/arduino-cli.yaml export ARDUINO_BOARD_MANAGER_ADDITIONAL_URLS ?= https://raw.githubusercontent.com/keyboardio/boardsmanager/master/package_keyboardio_index.json system_arduino_cli=$(shell command -v arduino-cli || true) ifeq ($(system_arduino_cli),) export ARDUINO_CLI ?= $(KALEIDOSCOPE_BIN_DIR)/arduino-cli else export ARDUINO_CLI ?= $(system_arduino_cli) endif ifneq ($(VERBOSE),) export ARDUINO_VERBOSE ?= --verbose else export ARDUINO_VERBOSE ?= endif # This is horrible. But because make doesn't really support # multi-line variables and we want to cache the full # _arduino_props in a variable, which means letting make # split the properties on space, which is what it converts # newlines into. To make this go, we we need to replace interior # spaces in the variables with something. We chose the fire # emoji, since it accurately represents our feelings on this # state of affairs. Later, when finding props, we need to reverse # this process, turning fire into space. _arduino_props = $(shell ${ARDUINO_CLI} compile --show-properties "$${SKETCH_FILE_PATH}"|perl -p -e"s/ /🔥/g") $(call make-lazy,_arduino_props) _arduino_prop = $(subst $1=,,$(subst 🔥, ,$(filter $1=%,$(_arduino_props)))) # How to use_arduino_prop # $(call _arduino_prop,recipe.hooks.sketch.prebuild.2.pattern) # # Build path config TMPDIR ?= /tmp export KALEIDOSCOPE_TEMP_PATH ?= $(TMPDIR)/kaleidoscope-$(USER) export KALEIDOSCOPE_BUILD_PATH ?= $(KALEIDOSCOPE_TEMP_PATH)/build export KALEIDOSCOPE_OUTPUT_PATH ?= $(KALEIDOSCOPE_TEMP_PATH)/output export CORE_CACHE_PATH ?= $(KALEIDOSCOPE_TEMP_PATH)/arduino-cores # If the sketch is defined ifneq ($(SKETCH),) # If the sketch isn't a directory, we want to get the directory the sketch is in ifeq ($(wildcard $(SKETCH)/.),) sketch_dir := $(dir $(SKETCH)) endif else # If the sketch wasn't defined as we came in, assume the current directory # is where we're looking sketch_dir := $(realpath $(CURDIR)) endif export SKETCH_BASE_NAME := $(notdir $(sketch_dir)) export SKETCH_FILE_NAME := $(addsuffix .ino, $(SKETCH_BASE_NAME)) sketch_dir_candidates = $(sketch_dir) src/ . sketch_exists_p = $(realpath $(wildcard $(dir)/$(SKETCH_FILE_NAME))) export FQBN := $(call _arduino_prop,build.fqbn) # Flashing related config port = $(shell $(ARDUINO_CLI) board list --format=text | grep $(FQBN) |cut -d' ' -f 1) flashing_instructions = $(call _arduino_prop,build.flashing_instructions) ifeq ($(flashing_instructions),) flashing_instructions = "If your keyboard needs you to do something to put it in flashing mode, do that now." endif export BOOTLOADER_PATH := $(call _arduino_prop,runtime.platform.path)/bootloaders/$(call _arduino_prop,bootloader.file) # Find the path of the sketch file export SKETCH_FILE_PATH := $(firstword $(foreach dir,$(sketch_dir_candidates),$(sketch_exists_p))) # We -could- check to see if sketch-dir is in git before running this command # but since we'd just return an empty value in that case, why bother? export GIT_VERSION := $(shell git -C "$(sketch_dir)" describe --abbrev=6 --dirty --alway 2>/dev/null || echo 'unknown') export SKETCH_IDENTIFIER := $(shell echo "$${SKETCH_FILE_PATH}" | cksum | cut -d ' ' -f 1)-$(SKETCH_FILE_NAME) export SKETCH_BUILD_DIR ?= $(SKETCH_IDENTIFIER)/build export SKETCH_OUTPUT_DIR ?= $(SKETCH_IDENTIFIER)/output export BUILD_PATH ?= $(KALEIDOSCOPE_BUILD_PATH)/$(SKETCH_BUILD_DIR) export OUTPUT_PATH ?= $(KALEIDOSCOPE_OUTPUT_PATH)/$(SKETCH_OUTPUT_DIR) export OUTPUT_FILE_PREFIX := $(SKETCH_BASE_NAME)-$(GIT_VERSION) export HEX_FILE_PATH := $(OUTPUT_PATH)/$(OUTPUT_FILE_PREFIX).hex export HEX_FILE_WITH_BOOTLOADER_PATH := $(OUTPUT_PATH)/$(OUTPUT_FILE_PREFIX)-with-bootloader.hex export ELF_FILE_PATH := $(OUTPUT_PATH)/$(OUTPUT_FILE_PREFIX).elf export LIB_FILE_PATH := $(OUTPUT_PATH)/$(OUTPUT_FILE_PREFIX).a export LIB_PROPERTIES_PATH := "../.." # We should use compiler.path instead of appending bin, but we # don't have substitution for arduino props yet export COMPILER_PATH := $(call _arduino_prop,runtime.tools.avr-gcc.path)/bin # Allow the compiler prefix to be empty for virtual builds COMPILER_PREFIX ?= avr- AVR_OBJDUMP := ${COMPILER_PATH}/${COMPILER_PREFIX}objdump AVR_OBJCOPY := ${COMPILER_PATH}/${COMPILER_PREFIX}objcopy AVR_NM := ${COMPILER_PATH}/${COMPILER_PREFIX}nm AVR_SIZE := ${COMPILER_PATH}/${COMPILER_PREFIX}size $(SKETCH_FILE_PATH): @echo "Sketch is $(SKETCH_FILE_PATH)" .PHONY: compile configure-arduino-cli install-arduino-core-kaleidoscope install-arduino-core-avr .PHONY: disassemble decompile size-map flash clean .DEFAULT_GOAL := compile all: @echo "Make all target doesn't do anything" @: ## Do not remove this line, otherwise `make all` will trigger the `%` rule too. install-arduino-cli: curl -fsSL https://raw.githubusercontent.com/arduino/arduino-cli/master/install.sh | BINDIR="$(KALEIDOSCOPE_BIN_DIR)" sh configure-arduino-cli: $(ARDUINO_CLI) config init install-arduino-core-kaleidoscope: $(ARDUINO_CLI) core install "keyboardio:avr" install-arduino-core-avr: $(ARDUINO_CLI) core install "arduino:avr" decompile: disassemble @: ## Do not remove this line, otherwise `make all` will trigger the `%` rule too. disassemble: ${ELF_FILE_PATH} ${AVR_OBJDUMP} -C -d "${ELF_FILE_PATH}" size-map: ${ELF_FILE_PATH} ${AVR_NM} --size-sort -C -r -l -t decimal "${ELF_FILE_PATH}" flash: ${HEX_FILE_PATH} ${ELF_FILE_PATH}: compile ${HEX_FILE_PATH}: compile hex-with-bootloader: ${HEX_FILE_PATH} awk '/^:00000001FF/ == 0' "${HEX_FILE_PATH}" >"${HEX_FILE_WITH_BOOTLOADER_PATH}" @echo "Using ${BOOTLOADER_PATH}" cat "${BOOTLOADER_PATH}" >>"${HEX_FILE_WITH_BOOTLOADER_PATH}" ln -sf -- "${OUTPUT_FILE_PREFIX}-with-bootloader.hex" "${OUTPUT_PATH}/${SKETCH_BASE_NAME}-latest-with-bootloader.hex" @echo Combined firmware and bootloader are now at @echo ${HEX_FILE_WITH_BOOTLOADER_PATH} @echo @echo Make sure you have the bootloader version you expect. @echo @echo @echo And TEST THIS ON REAL HARDWARE BEFORE YOU GIVE IT TO ANYONE. prop: @echo $(BOOTLOADER_PATH) @echo $(GIT_VERSION) clean: rm -rf -- "${OUTPUT_PATH}/*" _do_compile: compile: install -d "${OUTPUT_PATH}" @echo "Building ${SKETCH_FILE_PATH}" $(ARDUINO_CLI) compile \ --fqbn "${FQBN}" \ --libraries "${KALEIDOSCOPE_DIR}/.." \ --build-path "${BUILD_PATH}" \ --output-dir "${OUTPUT_PATH}" \ --build-cache-path "${CORE_CACHE_PATH}" \ --build-properties "compiler.cpp.extra_flags=${LOCAL_CFLAGS}" \ --warnings all ${ARDUINO_VERBOSE} \ "${SKETCH_FILE_PATH}" ifeq ($(LIBONLY),) cp "${BUILD_PATH}/${SKETCH_FILE_NAME}.hex" "${HEX_FILE_PATH}" cp "${BUILD_PATH}/${SKETCH_FILE_NAME}.elf" "${ELF_FILE_PATH}" ln -sf "${OUTPUT_FILE_PREFIX}.hex" "${OUTPUT_PATH}/${SKETCH_BASE_NAME}-latest.hex" ln -sf "${OUTPUT_FILE_PREFIX}.elf" "${OUTPUT_PATH}/${SKETCH_BASE_NAME}-latest.elf" else cp "${BUILD_PATH}/${SKETCH_FILE_NAME}.a" "${LIB_FILE_PATH}" ln -sf "${OUTPUT_FILE_PREFIX}.a" "${OUTPUT_PATH}/${SKETCH_BASE_NAME}-latest.a" endif $(info Build artifacts can be found in ${BUILD_PATH}) #TODO (arduino team) I'd love to do this with their json output #but it's short some of the data we kind of need flash: ifeq ($(port),) $(info Unable to detect keyboard serial port.) #@exit 1 endif $(info $(flashing_instructions)) $(info) $(info When you're ready to proceed, press 'Enter'.) $(info) @$(shell read) @$(ARDUINO_CLI) upload --fqbn $(FQBN) --port $(port) ${ARDUINO_VERBOSE}