# Select the board to build for: if not given on the command line, # then default to PYBV10. BOARD ?= arduino_zero ifeq ($(wildcard boards/$(BOARD)/.),) $(error Invalid BOARD specified) endif # If the build directory is not given, make it reflect the board name. BUILD ?= build-$(BOARD) include ../py/mkenv.mk -include mpconfigport.mk include boards/$(BOARD)/mpconfigboard.mk # qstr definitions (must come before including py.mk) # TODO(tannewt): Support friendly pin names like the stmhal implementations do. # Add $(BUILD)/pins_qstr.h $(BUILD)/modstm_qstr.h QSTR_DEFS = qstrdefsport.h # include py core make definitions include $(TOP)/py/py.mk CROSS_COMPILE = arm-none-eabi- BOSSAC := tools/bossac_osx HAL_DIR=hal/$(MCU_SERIES) INC += -I. INC += -I$(TOP) INC += -I$(TOP)/lib/mp-readline INC += -I$(TOP)/lib/timeutils INC += -Iasf_conf/ INC += -Iasf/common/boards/ INC += -Iasf/common/services/sleepmgr/ INC += -Iasf/common/services/storage/ctrl_access/ INC += -Iasf/common/services/usb/ INC += -Iasf/common/services/usb/class/cdc/ INC += -Iasf/common/services/usb/class/cdc/device/ INC += -Iasf/common/services/usb/class/hid/ INC += -Iasf/common/services/usb/class/hid/device/ INC += -Iasf/common/services/usb/class/hid/device/kbd/ INC += -Iasf/common/services/usb/class/hid/device/mouse/ INC += -Iasf/common/services/usb/class/msc/ INC += -Iasf/common/services/usb/class/msc/device/ INC += -Iasf/common/services/usb/udc/ INC += -Iasf/common/utils INC += -Iasf/common2/services/delay/ INC += $(addprefix -Iasf/sam0/,\ drivers/events \ drivers/extint \ drivers/port \ drivers/system \ drivers/adc/adc_sam_d_r \ drivers/dac \ drivers/dac/dac_sam_d_c \ drivers/sercom \ drivers/sercom/i2c \ drivers/system/clock \ drivers/system/clock/clock_samd21_r21_da \ drivers/system/interrupt \ drivers/system/interrupt/system_interrupt_samd21 \ drivers/system/pinmux \ drivers/system/power/power_sam_d_r \ drivers/system/reset/reset_sam_d_r \ drivers/tc \ drivers/usb \ utils \ utils/cmsis/samd21/include \ utils/cmsis/samd21/source \ utils/header_files \ utils/preprocessor \ ) INC += -Iasf/thirdparty/CMSIS/Include INC += -Iboards INC += -Iboards/$(BOARD)/ INC += -Ifreetouch/ INC += -I$(BUILD) CFLAGS_CORTEX_M0 = \ -mthumb \ -mabi=aapcs-linux \ -mcpu=cortex-m0plus \ -msoft-float \ -mfloat-abi=soft \ -fsingle-precision-constant \ -fno-strict-aliasing \ -Wdouble-promotion \ -Wno-endif-labels \ -Wstrict-prototypes \ -Werror-implicit-function-declaration \ -Wpointer-arith \ -Wfloat-equal \ -Wundef \ -Wshadow \ -Wwrite-strings \ -Wsign-compare \ -Wmissing-format-attribute \ -Wno-deprecated-declarations \ -Wpacked \ -Wnested-externs \ -Wunreachable-code \ -Wcast-align \ -D__$(CHIP_VARIANT)__ \ -DUSB_DEVICE_PRODUCT_ID=$(USB_PID) \ -DUSB_DEVICE_VENDOR_ID=$(USB_VID) \ -DBOARD=USER_BOARD \ -ffunction-sections \ -fdata-sections \ -fshort-enums \ -D ARM_MATH_CM0PLUS=true \ -DSYSTICK_MODE \ -DEXTINT_CALLBACK_MODE=true \ -DUDD_ENABLE \ -DUSART_CALLBACK_MODE=false \ -DSPI_CALLBACK_MODE=false \ -DI2C_MASTER_CALLBACK_MODE=false \ -DDAC_CALLBACK_MODE=false \ -DTCC_ASYNC=false \ -DADC_CALLBACK_MODE=false \ -DEVENTS_INTERRUPT_HOOKS_MODE=false \ -DI2S_CALLBACK_MODE=false \ -DTC_ASYNC=true \ -DUSB_DEVICE_LPM_SUPPORT \ -DCIRCUITPY_CANARY_WORD=0xADAF00 \ -DCIRCUITPY_SAFE_RESTART_WORD=0xDEADBEEF \ --param max-inline-insns-single=500 CFLAGS = $(INC) -Wall -Werror -std=gnu11 -nostdlib $(CFLAGS_CORTEX_M0) $(CFLAGS_MOD) $(COPT) #Debugging/Optimization ifeq ($(DEBUG), 1) # NDEBUG disables assert() statements. This reduces code size pretty dramatically, per tannewt. # Turn on Python modules useful for debugging (e.g. uheap, ustack). # -DMICROPY_DEBUG_MODULES may also be added to an -flto build, if you wish. CFLAGS += -Os -ggdb -DNDEBUG -DENABLE_MICRO_TRACE_BUFFER -DMICROPY_DEBUG_MODULES else # -finline-limit can shrink the image size. # -finline-limit=80 or so is similar to not having it on. # There is no simple default value, though. ifdef INTERNAL_FLASH_FILESYSTEM ## Not currently needed ## CFLAGS += -finline-limit=50 endif CFLAGS += -Os -DNDEBUG -flto $(GCC_INLINE_LIMIT) endif ifneq ($(FROZEN_DIR),) # To use frozen source modules, put your .py files in a subdirectory (eg scripts/) # and then invoke make with FROZEN_DIR=scripts (be sure to build from scratch). CFLAGS += -DMICROPY_MODULE_FROZEN_STR CFLAGS += -Wno-error=lto-type-mismatch endif # To use frozen bytecode, put your .py files in a subdirectory (eg frozen/) and # then invoke make with FROZEN_MPY_DIR=frozen or FROZEN_MPY_DIRS="dir1 dir2" # (be sure to build from scratch). ifneq ($(FROZEN_MPY_DIRS),) CFLAGS += -DMICROPY_QSTR_EXTRA_POOL=mp_qstr_frozen_const_pool CFLAGS += -DMICROPY_MODULE_FROZEN_MPY CFLAGS += -Wno-error=lto-type-mismatch endif LDFLAGS = $(CFLAGS) -nostartfiles -fshort-enums -mthumb -mcpu=cortex-m0plus -Lasf/thirdparty/CMSIS/Lib/GCC/ -Wl,-nostdlib -Wl,-T,$(LD_FILE) -Wl,-Map=$@.map -Wl,-cref -Wl,-gc-sections -specs=nano.specs LIBS = -larm_cortexM0l_math -lm -lgcc -lc # Use toolchain libm if we're not using our own. ifndef INTERNAL_LIBM LIBS += -lm endif SRC_ASF = $(addprefix asf/sam0/,\ drivers/adc/adc_sam_d_r/adc.c \ drivers/dac/dac_sam_d_c/dac.c \ drivers/dma/dma.c \ drivers/events/events_sam_d_r/events.c \ drivers/extint/extint_callback.c \ drivers/extint/extint_sam_d_r/extint.c \ drivers/i2s/i2s.c \ drivers/nvm/nvm.c \ drivers/port/port.c \ drivers/sercom/i2c/i2c_sam0/i2c_master.c \ drivers/sercom/sercom.c \ drivers/sercom/sercom_interrupt.c \ drivers/sercom/spi/spi.c \ drivers/sercom/usart/usart.c \ drivers/system/clock/clock_samd21_r21_da/clock.c \ drivers/system/clock/clock_samd21_r21_da/gclk.c \ drivers/system/interrupt/system_interrupt.c \ drivers/system/pinmux/pinmux.c \ drivers/system/system.c \ drivers/tc/tc_interrupt.c \ drivers/tc/tc_sam_d_r/tc.c \ drivers/tcc/tcc.c \ drivers/usb/stack_interface/usb_device_udd.c \ drivers/usb/stack_interface/usb_dual.c \ drivers/usb/usb_sam_d_r/usb.c \ ) SRC_C = \ access_vfs.c \ autoreload.c \ background.c \ fatfs_port.c \ flash_api.c \ main.c \ mphalport.c \ reset.c \ samd21_pins.c \ shared_dma.c \ rgb_led_status.c \ tick.c \ $(FLASH_IMPL) \ bindings/samd/__init__.c \ asf/common/services/sleepmgr/samd/sleepmgr.c \ asf/common/services/storage/ctrl_access/ctrl_access.c \ asf/common/services/usb/class/cdc/device/udi_cdc.c \ asf/common/services/usb/class/composite/device/udi_composite_desc.c \ asf/common/services/usb/class/hid/device/udi_hid.c \ asf/common/services/usb/class/hid/device/mouse/udi_hid_mouse.c \ asf/common/services/usb/class/hid/device/kbd/udi_hid_kbd.c \ asf/common/services/usb/class/msc/device/udi_msc.c \ asf/common/services/usb/udc/udc.c \ asf/common/utils/interrupt/interrupt_sam_nvic.c \ asf/common2/services/delay/sam0/systick_counter.c \ asf/sam0/utils/cmsis/samd21/source/gcc/startup_samd21.c \ asf/sam0/utils/cmsis/samd21/source/system_samd21.c \ asf/sam0/utils/syscalls/gcc/syscalls.c \ boards/$(BOARD)/board.c \ boards/$(BOARD)/pins.c \ freetouch/adafruit_ptc.c \ lib/oofatfs/ff.c \ lib/oofatfs/option/ccsbcs.c \ lib/timeutils/timeutils.c \ lib/utils/buffer_helper.c \ lib/utils/context_manager_helpers.c \ lib/utils/interrupt_char.c \ lib/utils/pyexec.c \ lib/utils/stdout_helpers.c \ lib/utils/sys_stdio_mphal.c \ lib/libc/string0.c \ lib/mp-readline/readline.c # Choose which flash filesystem impl to use. # (Right now INTERNAL_FLASH_FILESYSTEM and SPI_FLASH_FILESYSTEM are mutually exclusive. # But that might not be true in the future.) ifdef INTERNAL_FLASH_FILESYSTEM SRC_C += internal_flash.c endif ifdef SPI_FLASH_FILESYSTEM SRC_C += spi_flash.c endif SRC_COMMON_HAL = \ analogio/__init__.c \ analogio/AnalogIn.c \ analogio/AnalogOut.c \ audiobusio/__init__.c \ audiobusio/PDMIn.c \ audioio/__init__.c \ audioio/AudioOut.c \ board/__init__.c \ busio/__init__.c \ busio/I2C.c \ busio/SPI.c \ busio/UART.c \ digitalio/__init__.c \ digitalio/DigitalInOut.c \ microcontroller/__init__.c \ microcontroller/Pin.c \ microcontroller/Processor.c \ neopixel_write/__init__.c \ nvm/__init__.c \ nvm/ByteArray.c \ os/__init__.c \ pulseio/__init__.c \ pulseio/PulseIn.c \ pulseio/PulseOut.c \ pulseio/PWMOut.c \ storage/__init__.c \ time/__init__.c \ touchio/__init__.c \ touchio/TouchIn.c \ usb_hid/__init__.c \ usb_hid/Device.c ifdef INTERNAL_LIBM SRC_LIBM = $(addprefix lib/,\ libm/math.c \ libm/fmodf.c \ libm/nearbyintf.c \ libm/ef_sqrt.c \ libm/kf_rem_pio2.c \ libm/kf_sin.c \ libm/kf_cos.c \ libm/kf_tan.c \ libm/ef_rem_pio2.c \ libm/sf_sin.c \ libm/sf_cos.c \ libm/sf_tan.c \ libm/sf_frexp.c \ libm/sf_modf.c \ libm/sf_ldexp.c \ libm/asinfacosf.c \ libm/atanf.c \ libm/atan2f.c \ ) endif # These don't have corresponding files in each port but are still located in # shared-bindings to make it clear what the contents of the modules are. SRC_BINDINGS_ENUMS = \ digitalio/Direction.c \ digitalio/DriveMode.c \ digitalio/Pull.c \ help.c \ util.c SRC_COMMON_HAL_EXPANDED = $(addprefix shared-bindings/, $(SRC_COMMON_HAL)) \ $(addprefix shared-bindings/, $(SRC_BINDINGS_ENUMS)) \ $(addprefix common-hal/, $(SRC_COMMON_HAL)) SRC_SHARED_MODULE = \ bitbangio/__init__.c \ bitbangio/I2C.c \ bitbangio/OneWire.c \ bitbangio/SPI.c \ busio/OneWire.c \ gamepad/__init__.c \ gamepad/GamePad.c \ os/__init__.c \ random/__init__.c \ storage/__init__.c \ uheap/__init__.c \ ustack/__init__.c SRC_SHARED_MODULE_EXPANDED = $(addprefix shared-bindings/, $(SRC_SHARED_MODULE)) \ $(addprefix shared-module/, $(SRC_SHARED_MODULE)) OBJ = $(PY_O) $(addprefix $(BUILD)/, $(SRC_C:.c=.o)) OBJ += $(addprefix $(BUILD)/, $(SRC_ASF:.c=.o)) OBJ += $(addprefix $(BUILD)/, $(SRC_COMMON_HAL_EXPANDED:.c=.o)) OBJ += $(addprefix $(BUILD)/, $(SRC_SHARED_MODULE_EXPANDED:.c=.o)) ifdef INTERNAL_LIBM OBJ += $(addprefix $(BUILD)/, $(SRC_LIBM:.c=.o)) endif SRC_QSTR += $(SRC_C) $(SRC_COMMON_HAL_EXPANDED) $(SRC_SHARED_MODULE_EXPANDED) $(STM_SRC_C) all: $(BUILD)/firmware.bin $(BUILD)/firmware.uf2 $(BUILD)/firmware.elf: $(OBJ) $(STEPECHO) "LINK $@" $(Q)$(CC) -o $@ $(LDFLAGS) $^ -Wl,--start-group $(LIBS) -Wl,--end-group $(Q)$(SIZE) $@ | python3 $(TOP)/tools/build_memory_info.py $(LD_FILE) $(BUILD)/firmware.bin: $(BUILD)/firmware.elf $(ECHO) "Create $@" $(Q)$(OBJCOPY) -O binary -j .vectors -j .text -j .data $^ $@ $(BUILD)/firmware.uf2: $(BUILD)/firmware.bin $(ECHO) "Create $@" python2 $(TOP)/tools/uf2/utils/uf2conv.py -c -o $@ $^ deploy: $(BUILD)/firmware.bin $(ECHO) "Writing $< to the board" $(BOSSAC) -u $< # Run emulation build on a POSIX system with suitable terminal settings run: stty raw opost -echo build/firmware.elf @echo Resetting terminal... # This sleep is useful to spot segfaults sleep 1 reset test: $(BUILD)/firmware.elf $(Q)/bin/echo -e "print('hello world!', list(x+1 for x in range(10)), end='eol\\\\n')\\r\\n\\x04" | $(BUILD)/firmware.elf | tail -n2 | grep "^hello world! \\[1, 2, 3, 4, 5, 6, 7, 8, 9, 10\\]eol" include $(TOP)/py/mkrules.mk