1
#_______________________________________________________________________________
3
# edam's arduino makefile
4
#_______________________________________________________________________________
7
# Copyright (c) 2011 Tim Marston <tim@ed.am>.
9
# Permission is hereby granted, free of charge, to any person obtaining a copy
10
# of this software and associated documentation files (the "Software"), to deal
11
# in the Software without restriction, including without limitation the rights
12
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
13
# copies of the Software, and to permit persons to whom the Software is
14
# furnished to do so, subject to the following conditions:
16
# The above copyright notice and this permission notice shall be included in
17
# all copies or substantial portions of the Software.
19
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
20
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
21
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
22
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
23
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
24
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
27
#_______________________________________________________________________________
30
# This is a general purpose makefile for use with Arduino (see
31
# http://arduino.cc/) hardware and software. It works with the
32
# arduino-1.0 release and requires that to be downloaded separately.
33
# It can be downloaded from http://ed.am/dev/make/arduino-makefile
34
# where you can also find more information and documentation on it's
35
# use. The following text can only really be considered a reference
38
# To install, it is suggested that you keep arduino.mk somewhere and
39
# either symlink to it or include it in your make files. I keep mine
40
# at ~/src/arduino.mk. You will need to alter the value of the
41
# ARDUINODIR variable (below) to be the path where you have unpacked
42
# the arduino software from arduino.cc.
44
# This makefile can be used as a drop-in replacement for the Arduino
45
# IDE's build system. Simply create a symlink to it under the name
46
# "Makefile" and run make as described below (remembering to specify a
47
# BOARD). You would create the symlink like like so:
49
# $ ln -s ~/src/arduino.mk Makefile
51
# The presence of a .ino or .pde file causes the arduino.mk to
52
# atuomatically determine va;ues for SOURCES, TARGET and LIBRARIES.
53
# Any .c, .cc and .cpp files in the project directory (or a "util" or
54
# "utility" subdirectory) are automatically included in the build and
55
# are scanned for libraries that have been #included.
57
# Alternatively, if you want to manually specify build variables,
58
# create a Makefile that defines SOURCES and LIBRARARIES and then
59
# includes arduino.mk. (There is no need to define TARGET). You will
60
# also be expected to provide a main() function, for example in
61
# main.cc, which may or may not duplicate the functionality of the
62
# default main() that calls init() and loop(). Here is an example
65
# SOURCES := main.cc foo.cc
67
# include ~/src/arduino.mk
69
# A complete list of all the settings you can use in your Makefile
70
# follows shortly. It should be noted, however, that some variables
71
# are better specified in the environment or on the command line than
72
# in a Makefile. Specifically, the BOARD and SERIALDEV (if it is not
73
# automatically detected).
75
# When running make, you might want to specify the board type:
79
# Or in the environment:
81
# $ export BOARD=pro5v
84
# For a list of available board types, run `make boards`.
1
# Arduino 0022 Makefile
86
# Here is a list of all configuration parameters:
88
# BOARD Specify a target board type.
90
# SERIALDEV The unix device of the device where the arduino can be
91
# found. If unspecified, an attempt is made to determine
92
# the name of a connected arduino's serial device.
94
# The following configuration parameters can be determined automatically:
96
# TARGET The name of the target file. This need not be set if it
97
# is not determined automatically.
99
# SOURCES A list of all source files of whatever language. The
100
# language type is determined by the file extension.
102
# LIBRARIES A list of arduino libraries to build and include.
104
# This general-purpose makefile also defines the following goals for
105
# use on the command line when you run make:
107
# all This is the default if no goal is specified. It builds
108
# the target and uploads it.
110
# target Builds the target of your Makefile.
112
# upload Uploads the target to an attached arduino.
114
# clean Deletes temporary files.
116
# <file> Builds the specified file, either an object file or the
117
# target, from those that that would be built for the
120
#_______________________________________________________________________________
123
# The full path to the arduino software, from arduino.cc
124
ARDUINODIR := $(wildcard ~/opt/arduino-1.0)
126
# check arduino software
127
ifeq ($(wildcard $(ARDUINODIR)/hardware/arduino/boards.txt), )
128
$(error ARDUINODIR is not set correctly at the top of arduino.mk)
132
INOFILE := $(wildcard *.ino *.pde)
134
ifneq ($(words $(INOFILE)), 1)
135
$(error There is more than one .pde or .ino file in this directory!)
137
TARGET := $(basename $(INOFILE))
138
SOURCES := $(INOFILE) \
139
$(wildcard *.c *.cc *.cpp) \
140
$(wildcard $(addprefix util/, *.c *.cc *.cpp)) \
141
$(wildcard $(addprefix utility/, *.c *.cc *.cpp))
142
# automatically determine included libraries
143
ARDUINOLIBSAVAIL := $(notdir $(wildcard $(ARDUINODIR)/libraries/*))
144
LIBRARIES := $(filter $(ARDUINOLIBSAVAIL), \
145
$(shell sed -ne "s/^ *\# *include *[<\"]\(.*\)\.h[>\"]/\1/p" $(SOURCES)))
148
# no target? use default
153
# no serial device? attempt to detect an arduino
155
SERIALDEV := $(firstword $(wildcard /dev/ttyACM? /dev/ttyUSB?))
158
# no board? oh dear...
160
ifneq "$(MAKECMDGOALS)" "boards"
161
ifneq "$(MAKECMDGOALS)" "clean"
162
$(error BOARD is unset. Type 'make boards' to see possible values)
168
OBJECTS := $(addsuffix .o, $(basename $(SOURCES)))
169
ARDUINOSRCDIR := $(ARDUINODIR)/hardware/arduino/cores/arduino
170
ARDUINOLIB := _arduino.a
171
ARDUINOLIBTMP := _arduino.a.tmp
172
ARDUINOLIBOBJS := $(patsubst %, $(ARDUINOLIBTMP)/%.o, $(basename $(notdir \
173
$(wildcard $(addprefix $(ARDUINOSRCDIR)/, *.c *.cpp)))))
174
ARDUINOLIBOBJS += $(foreach lib, $(LIBRARIES), \
175
$(patsubst %, $(ARDUINOLIBTMP)/%.o, $(basename $(notdir \
176
$(wildcard $(addprefix $(ARDUINODIR)/libraries/$(lib)/, *.c *.cpp))))))
178
# obtain board parameters from the arduino boards.txt file
179
BOARDS_FILE := $(ARDUINODIR)/hardware/arduino/boards.txt
181
$(shell sed -ne "s/$(BOARD).build.mcu=\(.*\)/\1/p" $(BOARDS_FILE))
182
BOARD_BUILD_FCPU := \
183
$(shell sed -ne "s/$(BOARD).build.f_cpu=\(.*\)/\1/p" $(BOARDS_FILE))
184
BOARD_BUILD_VARIANT := \
185
$(shell sed -ne "s/$(BOARD).build.variant=\(.*\)/\1/p" $(BOARDS_FILE))
186
BOARD_UPLOAD_SPEED := \
187
$(shell sed -ne "s/$(BOARD).upload.speed=\(.*\)/\1/p" $(BOARDS_FILE))
188
BOARD_UPLOAD_PROTOCOL := \
189
$(shell sed -ne "s/$(BOARD).upload.protocol=\(.*\)/\1/p" $(BOARDS_FILE))
196
OBJCOPY := avr-objcopy
200
CPPFLAGS = -Os -Wall -fno-exceptions -ffunction-sections -fdata-sections
201
CPPFLAGS += -mmcu=$(BOARD_BUILD_MCU) -DF_CPU=$(BOARD_BUILD_FCPU)
202
CPPFLAGS += -I. -Iutil -Iutility -I$(ARDUINOSRCDIR)
203
CPPFLAGS += -I$(ARDUINODIR)/hardware/arduino/variants/$(BOARD_BUILD_VARIANT)/
204
CPPFLAGS += $(addprefix -I$(ARDUINODIR)/libraries/,$(LIBRARIES))
205
AVRDUDEFLAGS = -C $(ARDUINODIR)/hardware/tools/avrdude.conf -DV
206
AVRDUDEFLAGS += -p $(BOARD_BUILD_MCU) -P $(SERIALDEV)
207
AVRDUDEFLAGS += -c $(BOARD_UPLOAD_PROTOCOL) -b $(BOARD_UPLOAD_SPEED)
208
LINKFLAGS = -Os -Wl,--gc-sections -mmcu=$(BOARD_BUILD_MCU)
213
#_______________________________________________________________________________
216
.PHONY: all target upload clean boards
220
target: $(TARGET).hex
223
stty -F $(SERIALDEV) hupcl
224
$(AVRDUDE) $(AVRDUDEFLAGS) -U flash:w:$(TARGET).hex:i
4
# written by olikraus@gmail.com
7
# - boards.txt is used to derive parameters
8
# - All intermediate files are put into a separate directory (TMPDIRNAME)
9
# - Simple use: Copy Makefile into the same directory of the .pde file
12
# - requires UNIX environment
13
# - TMPDIRNAME must be subdirectory of the current directory.
16
# all build everything
17
# upload build and upload to arduino
18
# clean remove all temporary files (includes final hex file)
21
# 001 28 Apr 2010 first release
22
# 002 05 Oct 2010 added 'uno'
25
#=== user configuration ===
26
# All ...PATH variables must have a '/' at the end
28
# Board (and prozessor) information: see $(ARDUINO_PATH)hardware/arduino/boards.txt
32
# atmega328 Arduino Duemilanove or Nano w/ ATmega328
33
# diecimila Arduino Diecimila, Duemilanove, or Nano w/ ATmega168
36
# lilypad328 LilyPad Arduino w/ ATmega328
39
# additional (comma separated) defines
40
# -DDOGM128_HW board is connected to DOGM128 display
41
# -DDOGM132_HW board is connected to DOGM132 display
42
# -DDOGS102_HW board is connected to DOGS102 display
43
# -DDOG_REVERSE 180 degree rotation
46
# The location where the avr tools (e.g. avr-gcc) are located. Requires a '/' at the end.
47
# Can be empty if all tools are accessable through the search path
48
AVR_TOOLS_PATH:=/usr/bin/
50
# Install path of the arduino software. Requires a '/' at the end.
51
ARDUINO_PATH:=/home/edam/opt/arduino-0022/
53
# Install path for avrdude. Requires a '/' at the end. Can be empty if
54
# avrdude is in the search path.
55
AVRDUDE_PATH:=$(ARDUINO_PATH)hardware/tools/
57
# The unix device where we can reach the arduino board
59
# Duemilanove: /dev/ttyUSB0
60
AVRDUDE_PORT:=/dev/ttyACM0
62
# List of all libaries which should be included.
63
#EXTRA_DIRS+=$(ARDUINO_PATH)libraries/SPI/
64
#EXTRA_DIRS+=$(ARDUINO_PATH)libraries/Ethernet/
65
#EXTRA_DIRS+=$(ARDUINO_PATH)libraries/Ethernet/utility/
67
#=== fetch parameter from boards.txt processor parameter ===
68
# the basic idea is to get most of the information from boards.txt
70
BOARDS_TXT:=$(ARDUINO_PATH)hardware/arduino/boards.txt
72
# get the MCU value from the $(BOARD).build.mcu variable. For the
73
# atmega328 board this is atmega328p
74
MCU:=$(shell sed -n -e "s/$(BOARD).build.mcu=\(.*\)/\1/p" $(BOARDS_TXT))
75
# get the F_CPU value from the $(BOARD).build.f_cpu variable. For the
76
# atmega328 board this is 16000000
77
F_CPU:=$(shell sed -n -e "s/$(BOARD).build.f_cpu=\(.*\)/\1/p" $(BOARDS_TXT))
80
# get the AVRDUDE_UPLOAD_RATE value from the $(BOARD).upload.speed
81
# variable. For the atmega328 board this is 57600
82
AVRDUDE_UPLOAD_RATE:=$(shell sed -n -e "s/$(BOARD).upload.speed=\(.*\)/\1/p" $(BOARDS_TXT))
83
# get the AVRDUDE_PROGRAMMER value from the $(BOARD).upload.protocol
84
# variable. For the atmega328 board this is stk500
85
#AVRDUDE_PROGRAMMER:=$(shell sed -n -e "s/$(BOARD).upload.protocol=\(.*\)/\1/p" $(BOARDS_TXT))
86
# use stk500v1, because stk500 will default to stk500v2
87
AVRDUDE_PROGRAMMER:=stk500v1
89
#=== identify user files ===
90
PDESRC:=$(shell ls *.pde)
91
TARGETNAME=$(basename $(PDESRC))
93
CDIRS:=$(EXTRA_DIRS) $(addsuffix utility/,$(EXTRA_DIRS))
94
CDIRS:=*.c utility/*.c $(addsuffix *.c,$(CDIRS)) $(ARDUINO_PATH)hardware/arduino/cores/arduino/*.c
95
CSRC:=$(shell ls $(CDIRS) 2>/dev/null)
97
CCSRC:=$(shell ls *.cc 2>/dev/null)
99
CPPDIRS:=$(EXTRA_DIRS) $(addsuffix utility/,$(EXTRA_DIRS))
100
CPPDIRS:=*.cpp utility/*.cpp $(addsuffix *.cpp,$(CPPDIRS)) $(ARDUINO_PATH)hardware/arduino/cores/arduino/*.cpp
101
CPPSRC:=$(shell ls $(CPPDIRS) 2>/dev/null)
103
#=== build internal variables ===
105
# the name of the subdirectory where everything is stored
107
TMPDIRPATH:=$(TMPDIRNAME)/
109
AVRTOOLSPATH:=$(AVR_TOOLS_PATH)
111
OBJCOPY:=$(AVRTOOLSPATH)avr-objcopy
112
OBJDUMP:=$(AVRTOOLSPATH)avr-objdump
113
SIZE:=$(AVRTOOLSPATH)avr-size
115
CPPSRC:=$(addprefix $(TMPDIRPATH),$(PDESRC:.pde=.cpp)) $(CPPSRC)
118
CCOBJ:=$(CCSRC:.cc=.o)
119
CPPOBJ:=$(CPPSRC:.cpp=.o)
121
OBJFILES:=$(COBJ) $(CCOBJ) $(CPPOBJ)
122
DIRS:= $(dir $(OBJFILES))
124
DEPFILES:=$(OBJFILES:.o=.d)
125
# assembler files from avr-gcc -S
126
ASSFILES:=$(OBJFILES:.o=.s)
127
# disassembled object files with avr-objdump -S
128
DISFILES:=$(OBJFILES:.o=.dis)
131
LIBNAME:=$(TMPDIRPATH)$(TARGETNAME).a
132
ELFNAME:=$(TMPDIRPATH)$(TARGETNAME).elf
133
HEXNAME:=$(TMPDIRPATH)$(TARGETNAME).hex
135
AVRDUDE_FLAGS = -V -F
136
AVRDUDE_FLAGS += -C $(ARDUINO_PATH)/hardware/tools/avrdude.conf
137
AVRDUDE_FLAGS += -p $(MCU)
138
AVRDUDE_FLAGS += -P $(AVRDUDE_PORT)
139
AVRDUDE_FLAGS += -c $(AVRDUDE_PROGRAMMER)
140
AVRDUDE_FLAGS += -b $(AVRDUDE_UPLOAD_RATE)
141
AVRDUDE_FLAGS += -U flash:w:$(HEXNAME)
143
AVRDUDE = $(AVRDUDE_PATH)avrdude
145
#=== predefined variable override ===
147
# use "make -p -f/dev/null" to see the default rules and definitions
148
# Build C and C++ flags. Include path information must be placed here
149
COMMON_FLAGS = -DF_CPU=$(F_CPU) -mmcu=$(MCU) $(DEFS)
150
# COMMON_FLAGS += -gdwarf-2
152
COMMON_FLAGS += -Wall -funsigned-char -funsigned-bitfields -fpack-struct -fshort-enums
153
COMMON_FLAGS += -I. -Iutility
154
COMMON_FLAGS += -I$(ARDUINO_PATH)hardware/arduino/cores/arduino
155
COMMON_FLAGS += $(addprefix -I,$(EXTRA_DIRS))
156
COMMON_FLAGS += -fno-exceptions -ffunction-sections -fdata-sections -Wl,--gc-sections
157
COMMON_FLAGS += -Wl,--relax
158
COMMON_FLAGS += -mcall-prologues
160
CFLAGS = $(COMMON_FLAGS) -std=gnu99 -Wstrict-prototypes
161
CXXFLAGS = $(COMMON_FLAGS)
163
# Replace standard build tools by avr tools
164
CC = $(AVRTOOLSPATH)avr-gcc
165
CXX = $(AVRTOOLSPATH)avr-g++
166
AR = @$(AVRTOOLSPATH)avr-ar
169
# "rm" must be able to delete a directory tree
174
# add rules for the C/C++ files where the .o file is placed in the
175
# TMPDIRPATH reuse existing variables as far as possible
177
$(TMPDIRPATH)%.o: %.c
179
@$(COMPILE.c) $(OUTPUT_OPTION) $<
181
$(TMPDIRPATH)%.o: %.cc
183
@$(COMPILE.cc) $(OUTPUT_OPTION) $<
185
$(TMPDIRPATH)%.o: %.cpp
187
@$(COMPILE.cpp) $(OUTPUT_OPTION) $<
189
$(TMPDIRPATH)%.s: %.c
190
@$(COMPILE.c) $(OUTPUT_OPTION) -S $<
192
$(TMPDIRPATH)%.s: %.cc
193
@$(COMPILE.cc) $(OUTPUT_OPTION) -S $<
195
$(TMPDIRPATH)%.s: %.cpp
196
@$(COMPILE.cpp) $(OUTPUT_OPTION) -S $<
198
$(TMPDIRPATH)%.dis: $(TMPDIRPATH)%.o
199
@$(OBJDUMP) -S $< > $@
201
.SUFFIXES: .elf .hex .pde
204
@$(OBJCOPY) -O ihex -R .eeprom $< $@
206
$(TMPDIRPATH)%.cpp: %.pde
207
@cat $(ARDUINO_PATH)hardware/arduino/cores/arduino/main.cpp > $@
210
@echo 'extern "C" void __cxa_pure_virtual() { while (1); }' >> $@
214
all: tmpdir $(HEXNAME) assemblersource showsize
215
ls -al $(HEXNAME) $(ELFNAME)
217
$(ELFNAME): $(LIBNAME)($(addprefix $(TMPDIRPATH),$(OBJFILES)))
218
$(LINK.o) $(COMMON_FLAGS) $(LIBNAME) $(LOADLIBES) $(LDLIBS) -o $@
220
$(LIBNAME)(): $(addprefix $(TMPDIRPATH),$(OBJFILES))
222
#=== create temp directory ===
223
# not really required, because it will be also created during the
224
# dependency handling
227
@test -d $(TMPDIRPATH) || mkdir $(TMPDIRPATH)
229
#=== create assembler files for each C/C++ file ===
230
.PHONY: assemblersource
231
assemblersource: $(addprefix $(TMPDIRPATH),$(ASSFILES)) $(addprefix $(TMPDIRPATH),$(DISFILES))
234
#=== show the section sizes of the ELF file ===
239
#=== clean up target ===
240
# this is simple: the TMPDIRPATH is removed
228
rm -f $(TARGET).elf $(TARGET).hex $(ARDUINOLIB) *~
229
rm -rf $(ARDUINOLIBTMP)
232
@echo Available values for BOARD:
233
@sed -ne '/^#/d;s/^\(.*\).name=\(.*\)/\1 \2/;T' \
234
-e 's/\(.\{12\}\) *\(.*\)/\1 \2/;p' $(BOARDS_FILE)
236
# building the target
238
$(TARGET).hex: $(TARGET).elf
239
$(OBJCOPY) -O ihex -R .eeprom $< $@
241
.INTERMEDIATE: $(TARGET).elf
243
$(TARGET).elf: $(ARDUINOLIB) $(OBJECTS)
244
$(CC) $(LINKFLAGS) $(OBJECTS) $(ARDUINOLIB) -o $@
247
$(COMPILE.cpp) -o $@ -x c++ -include $(ARDUINOSRCDIR)/Arduino.h $<
250
$(COMPILE.cpp) -o $@ -x c++ -include $(ARDUINOSRCDIR)/Arduino.h $<
252
# building the arduino library
254
$(ARDUINOLIB): $(ARDUINOLIBOBJS)
256
rm -rf $(ARDUINOLIBTMP)
258
.INTERMEDIATE: $(ARDUINOLIBOBJS)
260
$(ARDUINOLIBTMP)/%.o: $(ARDUINOSRCDIR)/%.c
261
@test -d $(ARDUINOLIBTMP) || mkdir $(ARDUINOLIBTMP)
262
$(COMPILE.c) -o $@ $<
264
$(ARDUINOLIBTMP)/%.o: $(ARDUINOSRCDIR)/%.cpp
265
@test -d $(ARDUINOLIBTMP) || mkdir $(ARDUINOLIBTMP)
266
$(COMPILE.cpp) -o $@ $<
268
$(ARDUINOLIBTMP)/%.o: $(ARDUINODIR)/libraries/*/%.c
269
@test -d $(ARDUINOLIBTMP) || mkdir $(ARDUINOLIBTMP)
270
$(COMPILE.c) -o $@ $<
272
$(ARDUINOLIBTMP)/%.o: $(ARDUINODIR)/libraries/*/%.cpp
273
@test -d $(ARDUINOLIBTMP) || mkdir $(ARDUINOLIBTMP)
274
$(COMPILE.cpp) -o $@ $<
245
# Program the device.
246
# step 1: reset the arduino board with the stty command
247
# step 2: user avrdude to upload the software
250
stty -F $(AVRDUDE_PORT) hupcl
251
$(AVRDUDE) $(AVRDUDE_FLAGS)
254
# === dependency handling ===
255
# From the gnu make manual (section 4.14, Generating Prerequisites
256
# Automatically) Additionally (because this will be the first executed
257
# rule) TMPDIRPATH is created here. Instead of "sed" the "echo"
259
# cd $(TMPDIRPATH); mkdir -p $(DIRS) 2> /dev/null; cd ..
260
DEPACTION=test -d $(TMPDIRPATH) || mkdir $(TMPDIRPATH);\
261
mkdir -p $(addprefix $(TMPDIRPATH),$(DIRS));\
262
set -e; echo -n $@ $(dir $@) > $@; $(CC) -MM $(COMMON_FLAGS) $< >> $@
265
$(TMPDIRPATH)%.d: %.c
268
$(TMPDIRPATH)%.d: %.cc
272
$(TMPDIRPATH)%.d: %.cpp
275
# Include dependency files. If a .d file is missing, a warning is
276
# created and the .d file is created This warning is not a problem
277
# (gnu make manual, section 3.3 Including Other Makefiles)
278
-include $(addprefix $(TMPDIRPATH),$(DEPFILES))