Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 1 | #---------------------------------------------------------------------- |
| 2 | # Clients fill in the source files to build |
| 3 | #---------------------------------------------------------------------- |
| 4 | # C_SOURCES := main.c |
| 5 | # CXX_SOURCES := |
| 6 | # OBJC_SOURCES := |
| 7 | # OBJCXX_SOURCES := |
| 8 | # DYLIB_C_SOURCES := |
Johnny Chen | e9f9770 | 2012-01-11 01:42:58 +0000 | [diff] [blame] | 9 | # DYLIB_CXX_SOURCES := |
| 10 | # |
| 11 | # Specifying DYLIB_ONLY has the effect of building dylib only, skipping |
| 12 | # the building of the a.out executable program. For example, |
| 13 | # DYLIB_ONLY := YES |
| 14 | # |
Johnny Chen | 4876b5f | 2012-01-11 01:59:55 +0000 | [diff] [blame] | 15 | # And also might be of interest: |
| 16 | # FRAMEWORK_INCLUDES (Darwin only) := |
| 17 | # CFLAGS_EXTRAS := |
| 18 | # LD_EXTRAS := |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 19 | |
| 20 | # Uncomment line below for debugging shell commands |
| 21 | # SHELL = /bin/sh -x |
| 22 | |
| 23 | #---------------------------------------------------------------------- |
Johnny Chen | 6c17c08 | 2010-09-16 20:54:06 +0000 | [diff] [blame] | 24 | # If ARCH is not defined, default to x86_64. |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 25 | # If OS is not defined, use 'uname -s' to determine the OS name. |
Johnny Chen | 6c17c08 | 2010-09-16 20:54:06 +0000 | [diff] [blame] | 26 | #---------------------------------------------------------------------- |
| 27 | ifeq "$(ARCH)" "" |
| 28 | ARCH = x86_64 |
| 29 | endif |
| 30 | |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 31 | ifeq "$(OS)" "" |
| 32 | OS = $(shell uname -s) |
| 33 | endif |
| 34 | |
Johnny Chen | 6c17c08 | 2010-09-16 20:54:06 +0000 | [diff] [blame] | 35 | #---------------------------------------------------------------------- |
Johnny Chen | 597cbbb | 2011-08-23 21:54:10 +0000 | [diff] [blame] | 36 | # CC defaults to clang. |
Johnny Chen | 4ab4a9b | 2011-08-24 18:12:53 +0000 | [diff] [blame] | 37 | # |
| 38 | # If you change the defaults of CC, be sure to also change it in the file |
| 39 | # test/plugins/builder_base.py, which provides a Python way to return the |
| 40 | # value of the make variable CC -- getCompiler(). |
| 41 | # |
Johnny Chen | 6055b44 | 2011-01-14 20:55:13 +0000 | [diff] [blame] | 42 | # See also these functions: |
| 43 | # o cxx_compiler |
| 44 | # o cxx_linker |
Johnny Chen | d43e208 | 2011-01-14 20:46:49 +0000 | [diff] [blame] | 45 | #---------------------------------------------------------------------- |
Johnny Chen | 597cbbb | 2011-08-23 21:54:10 +0000 | [diff] [blame] | 46 | CC ?= clang |
Johnny Chen | d43e208 | 2011-01-14 20:46:49 +0000 | [diff] [blame] | 47 | ifeq "$(CC)" "cc" |
Johnny Chen | 597cbbb | 2011-08-23 21:54:10 +0000 | [diff] [blame] | 48 | CC = clang |
Johnny Chen | d43e208 | 2011-01-14 20:46:49 +0000 | [diff] [blame] | 49 | endif |
| 50 | |
| 51 | #---------------------------------------------------------------------- |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 52 | # Change any build/tool options needed |
| 53 | #---------------------------------------------------------------------- |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 54 | CFLAGS ?= -gdwarf-2 -O0 |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 55 | CFLAGS += $(FRAMEWORK_INCLUDES) $(CFLAGS_EXTRAS) |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 56 | ifeq "$(OS)" "Darwin" |
| 57 | CFLAGS += -arch $(ARCH) |
Greg Clayton | 3bffb08 | 2011-12-10 02:15:28 +0000 | [diff] [blame] | 58 | DS := dsymutil |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 59 | DSFLAGS = |
| 60 | DSYM = $(EXE).dSYM |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 61 | AR := libtool |
| 62 | ARFLAGS := -static -o |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 63 | endif |
| 64 | |
Johnny Chen | 53e2edb | 2011-08-09 20:07:16 +0000 | [diff] [blame] | 65 | CXXFLAGS +=$(CFLAGS) |
| 66 | LD = $(CC) |
| 67 | LDFLAGS ?= $(CFLAGS) |
| 68 | LDFLAGS += $(LD_EXTRAS) |
| 69 | OBJECTS = |
| 70 | EXE ?= a.out |
| 71 | |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 72 | ifneq "$(DYLIB_NAME)" "" |
| 73 | ifeq "$(OS)" "Darwin" |
| 74 | DYLIB_FILENAME = lib$(DYLIB_NAME).dylib |
| 75 | else |
| 76 | DYLIB_FILENAME = lib$(DYLIB_NAME).so |
| 77 | endif |
| 78 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 79 | |
Johnny Chen | bdb4efc | 2011-01-14 18:19:53 +0000 | [diff] [blame] | 80 | # Function that returns the counterpart C++ compiler, given $(CC) as arg. |
| 81 | cxx_compiler = $(if $(findstring clang,$(1)), $(subst clang,clang++,$(1)), $(if $(findstring llvm-gcc,$(1)), $(subst llvm-gcc,llvm-g++,$(1)), $(subst gcc,g++,$(1)))) |
| 82 | |
| 83 | # Function that returns the C++ linker, given $(CC) as arg. |
Greg Clayton | d50d238 | 2012-01-10 00:00:15 +0000 | [diff] [blame] | 84 | cxx_linker = $(if $(findstring clang,$(1)), $(subst clang,clang++,$(1)), $(if $(findstring llvm-gcc,$(1)), $(subst llvm-gcc,llvm-g++,$(1)), $(subst gcc,g++,$(1)))) |
Johnny Chen | 832d233 | 2010-09-30 01:22:34 +0000 | [diff] [blame] | 85 | |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 86 | #---------------------------------------------------------------------- |
| 87 | # dylib settings |
| 88 | #---------------------------------------------------------------------- |
| 89 | ifneq "$(strip $(DYLIB_C_SOURCES))" "" |
| 90 | DYLIB_OBJECTS +=$(strip $(DYLIB_C_SOURCES:.c=.o)) |
| 91 | endif |
| 92 | |
Greg Clayton | d50d238 | 2012-01-10 00:00:15 +0000 | [diff] [blame] | 93 | ifneq "$(strip $(DYLIB_CXX_SOURCES))" "" |
| 94 | DYLIB_OBJECTS +=$(strip $(DYLIB_CXX_SOURCES:.cpp=.o)) |
| 95 | CXX = $(call cxx_compiler,$(CC)) |
| 96 | LD = $(call cxx_linker,$(CC)) |
| 97 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 98 | |
| 99 | #---------------------------------------------------------------------- |
| 100 | # Check if we have any C source files |
| 101 | #---------------------------------------------------------------------- |
| 102 | ifneq "$(strip $(C_SOURCES))" "" |
| 103 | OBJECTS +=$(strip $(C_SOURCES:.c=.o)) |
| 104 | endif |
| 105 | |
| 106 | #---------------------------------------------------------------------- |
| 107 | # Check if we have any C++ source files |
| 108 | #---------------------------------------------------------------------- |
| 109 | ifneq "$(strip $(CXX_SOURCES))" "" |
| 110 | OBJECTS +=$(strip $(CXX_SOURCES:.cpp=.o)) |
Johnny Chen | 832d233 | 2010-09-30 01:22:34 +0000 | [diff] [blame] | 111 | CXX = $(call cxx_compiler,$(CC)) |
Johnny Chen | bdb4efc | 2011-01-14 18:19:53 +0000 | [diff] [blame] | 112 | LD = $(call cxx_linker,$(CC)) |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 113 | endif |
| 114 | |
| 115 | #---------------------------------------------------------------------- |
| 116 | # Check if we have any ObjC source files |
| 117 | #---------------------------------------------------------------------- |
| 118 | ifneq "$(strip $(OBJC_SOURCES))" "" |
| 119 | OBJECTS +=$(strip $(OBJC_SOURCES:.m=.o)) |
| 120 | LDFLAGS +=-lobjc |
| 121 | endif |
| 122 | |
| 123 | #---------------------------------------------------------------------- |
| 124 | # Check if we have any ObjC++ source files |
| 125 | #---------------------------------------------------------------------- |
| 126 | ifneq "$(strip $(OBJCXX_SOURCES))" "" |
| 127 | OBJECTS +=$(strip $(OBJCXX_SOURCES:.mm=.o)) |
Johnny Chen | 832d233 | 2010-09-30 01:22:34 +0000 | [diff] [blame] | 128 | CXX = $(call cxx_compiler,$(CC)) |
Johnny Chen | bdb4efc | 2011-01-14 18:19:53 +0000 | [diff] [blame] | 129 | LD = $(call cxx_linker,$(CC)) |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 130 | ifeq $(findstring lobjc,$(LDFLAGS)) "" |
| 131 | LDFLAGS +=-lobjc |
| 132 | endif |
| 133 | endif |
| 134 | |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 135 | #---------------------------------------------------------------------- |
| 136 | # Check if we have any C source files for archive |
| 137 | #---------------------------------------------------------------------- |
| 138 | ifneq "$(strip $(ARCHIVE_C_SOURCES))" "" |
| 139 | ARCHIVE_OBJECTS +=$(strip $(ARCHIVE_C_SOURCES:.c=.o)) |
| 140 | endif |
| 141 | |
| 142 | #---------------------------------------------------------------------- |
| 143 | # Check if we have any C++ source files for archive |
| 144 | #---------------------------------------------------------------------- |
| 145 | ifneq "$(strip $(ARCHIVE_CXX_SOURCES))" "" |
| 146 | ARCHIVE_OBJECTS +=$(strip $(ARCHIVE_CXX_SOURCES:.cpp=.o)) |
| 147 | CXX = $(call cxx_compiler,$(CC)) |
| 148 | LD = $(call cxx_linker,$(CC)) |
| 149 | endif |
| 150 | |
| 151 | #---------------------------------------------------------------------- |
| 152 | # Check if we have any ObjC source files for archive |
| 153 | #---------------------------------------------------------------------- |
| 154 | ifneq "$(strip $(ARCHIVE_OBJC_SOURCES))" "" |
| 155 | ARCHIVE_OBJECTS +=$(strip $(ARCHIVE_OBJC_SOURCES:.m=.o)) |
| 156 | LDFLAGS +=-lobjc |
| 157 | endif |
| 158 | |
| 159 | #---------------------------------------------------------------------- |
| 160 | # Check if we have any ObjC++ source files for archive |
| 161 | #---------------------------------------------------------------------- |
| 162 | ifneq "$(strip $(ARCHIVE_OBJCXX_SOURCES))" "" |
| 163 | ARCHIVE_OBJECTS +=$(strip $(ARCHIVE_OBJCXX_SOURCES:.mm=.o)) |
| 164 | CXX = $(call cxx_compiler,$(CC)) |
| 165 | LD = $(call cxx_linker,$(CC)) |
| 166 | ifeq $(findstring lobjc,$(LDFLAGS)) "" |
| 167 | LDFLAGS +=-lobjc |
| 168 | endif |
| 169 | endif |
| 170 | |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 171 | |
| 172 | #---------------------------------------------------------------------- |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 173 | # DYLIB_ONLY variable can be used to skip the building of a.out. |
| 174 | # See the sections below regarding dSYM file as well as the building of |
| 175 | # EXE from all the objects. |
| 176 | #---------------------------------------------------------------------- |
| 177 | |
| 178 | #---------------------------------------------------------------------- |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 179 | # Make the dSYM file from the executable if $(MAKE_DSYM) != "NO" |
| 180 | #---------------------------------------------------------------------- |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 181 | ifeq "$(OS)" "Darwin" |
Johnny Chen | 9101639 | 2011-06-20 20:08:26 +0000 | [diff] [blame] | 182 | ifneq "$(MAKE_DSYM)" "NO" |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 183 | ifeq "$(DYLIB_ONLY)" "" |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 184 | $(DSYM) : $(EXE) |
| 185 | $(DS) $(DSFLAGS) -o "$(DSYM)" "$(EXE)" |
Johnny Chen | 9101639 | 2011-06-20 20:08:26 +0000 | [diff] [blame] | 186 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 187 | endif |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 188 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 189 | |
| 190 | #---------------------------------------------------------------------- |
| 191 | # Compile the executable from all the objects. |
| 192 | #---------------------------------------------------------------------- |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 193 | ifneq "$(DYLIB_NAME)" "" |
| 194 | ifeq "$(DYLIB_ONLY)" "" |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 195 | $(EXE) : $(OBJECTS) $(ARCHIVE_NAME) $(DYLIB_FILENAME) |
| 196 | $(LD) $(LDFLAGS) $(OBJECTS) $(ARCHIVE_NAME) -L. -l$(DYLIB_NAME) -o "$(EXE)" |
Johnny Chen | beac8f9 | 2012-01-10 00:41:11 +0000 | [diff] [blame] | 197 | else |
| 198 | EXE = $(DYLIB_FILENAME) |
| 199 | endif |
| 200 | else |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 201 | $(EXE) : $(OBJECTS) $(ARCHIVE_NAME) |
| 202 | $(LD) $(LDFLAGS) $(OBJECTS) $(ARCHIVE_NAME) -o "$(EXE)" |
| 203 | endif |
| 204 | |
| 205 | #---------------------------------------------------------------------- |
| 206 | # Make the archive |
| 207 | #---------------------------------------------------------------------- |
| 208 | ifneq "$(ARCHIVE_NAME)" "" |
| 209 | ifeq "$(OS)" "Darwin" |
| 210 | $(ARCHIVE_NAME) : $(ARCHIVE_OBJECTS) |
| 211 | $(AR) $(ARFLAGS) $(ARCHIVE_NAME) $(ARCHIVE_OBJECTS) |
| 212 | $(RM) $(ARCHIVE_OBJECTS) |
| 213 | else |
| 214 | $(ARCHIVE_NAME) : $(foreach ar_obj,$(ARCHIVE_OBJECTS),$(ARCHIVE_NAME)($(ar_obj))) |
| 215 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 216 | endif |
| 217 | |
| 218 | #---------------------------------------------------------------------- |
| 219 | # Make the dylib |
| 220 | #---------------------------------------------------------------------- |
Peter Collingbourne | 518f03d | 2011-06-20 19:06:04 +0000 | [diff] [blame] | 221 | $(DYLIB_FILENAME) : $(DYLIB_OBJECTS) |
| 222 | ifeq "$(OS)" "Darwin" |
| 223 | $(LD) $(LDFLAGS) $(DYLIB_OBJECTS) -install_name "@executable_path/$(DYLIB_FILENAME)" -dynamiclib -o "$(DYLIB_FILENAME)" |
| 224 | else |
| 225 | $(LD) $(LDFLAGS) $(DYLIB_OBJECTS) -shared -o "$(DYLIB_FILENAME)" |
| 226 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 227 | |
| 228 | #---------------------------------------------------------------------- |
| 229 | # Automatic variables based on items already entered. Below we create |
| 230 | # an objects lists from the list of sources by replacing all entries |
| 231 | # that end with .c with .o, and we also create a list of prerequisite |
| 232 | # files by replacing all .c files with .d. |
| 233 | #---------------------------------------------------------------------- |
| 234 | PREREQS := $(OBJECTS:.o=.d) |
Johnny Chen | e3dc0f0 | 2010-08-24 16:35:00 +0000 | [diff] [blame] | 235 | ifneq "$(DYLIB_NAME)" "" |
| 236 | DYLIB_PREREQS := $(DYLIB_OBJECTS:.o=.d) |
| 237 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 238 | |
| 239 | #---------------------------------------------------------------------- |
| 240 | # Rule for Generating Prerequisites Automatically using .d files and |
| 241 | # the compiler -MM option. The -M option will list all system headers, |
| 242 | # and the -MM option will list all non-system dependencies. |
| 243 | #---------------------------------------------------------------------- |
| 244 | %.d: %.c |
| 245 | @set -e; rm -f $@; \ |
Johnny Chen | 189bff1 | 2011-08-04 20:44:56 +0000 | [diff] [blame] | 246 | $(CC) -M $(CFLAGS) $< > $@.$$$$; \ |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 247 | sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ |
| 248 | rm -f $@.$$$$ |
| 249 | |
| 250 | %.d: %.cpp |
| 251 | @set -e; rm -f $@; \ |
Johnny Chen | 189bff1 | 2011-08-04 20:44:56 +0000 | [diff] [blame] | 252 | $(CC) -M $(CXXFLAGS) $< > $@.$$$$; \ |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 253 | sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ |
| 254 | rm -f $@.$$$$ |
| 255 | |
| 256 | %.d: %.m |
| 257 | @set -e; rm -f $@; \ |
Johnny Chen | 189bff1 | 2011-08-04 20:44:56 +0000 | [diff] [blame] | 258 | $(CC) -M $(CFLAGS) $< > $@.$$$$; \ |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 259 | sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ |
| 260 | rm -f $@.$$$$ |
| 261 | |
| 262 | %.d: %.mm |
| 263 | @set -e; rm -f $@; \ |
Johnny Chen | 189bff1 | 2011-08-04 20:44:56 +0000 | [diff] [blame] | 264 | $(CC) -M $(CXXFLAGS) $< > $@.$$$$; \ |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 265 | sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; \ |
| 266 | rm -f $@.$$$$ |
| 267 | |
| 268 | #---------------------------------------------------------------------- |
| 269 | # Include all of the makefiles for each source file so we don't have |
| 270 | # to manually track all of the prerequisites for each source file. |
| 271 | #---------------------------------------------------------------------- |
| 272 | sinclude $(PREREQS) |
Johnny Chen | e3dc0f0 | 2010-08-24 16:35:00 +0000 | [diff] [blame] | 273 | ifneq "$(DYLIB_NAME)" "" |
| 274 | sinclude $(DYLIB_PREREQS) |
| 275 | endif |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 276 | |
| 277 | .PHONY: clean |
| 278 | dsym: $(DSYM) |
| 279 | all: $(EXE) $(DSYM) |
Johnny Chen | e1030cd | 2010-09-27 20:44:46 +0000 | [diff] [blame] | 280 | clean:: |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 281 | ifeq "$(DYLIB_NAME)" "" |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 282 | rm -rf "$(EXE)" "$(DSYM)" $(OBJECTS) $(PREREQS) $(ARCHIVE_NAME) $(ARCHIVE_OBJECTS) |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 283 | else |
Johnny Chen | fe14162 | 2012-01-12 23:09:42 +0000 | [diff] [blame^] | 284 | rm -rf "$(EXE)" "$(DSYM)" $(OBJECTS) $(PREREQS) $(ARCHIVE_NAME) $(ARCHIVE_OBJECTS) $(DYLIB_OBJECTS) $(DYLIB_PREREQS) $(DYLIB_FILENAME) $(DYLIB_FILENAME).dSYM |
Johnny Chen | 9bc867a | 2010-08-23 23:56:08 +0000 | [diff] [blame] | 285 | endif |
Johnny Chen | fa38041 | 2011-01-14 21:18:12 +0000 | [diff] [blame] | 286 | |
| 287 | #---------------------------------------------------------------------- |
| 288 | # From http://blog.melski.net/tag/debugging-makefiles/ |
| 289 | # |
| 290 | # Usage: make print-CC print-CXX print-LD |
| 291 | #---------------------------------------------------------------------- |
| 292 | print-%: |
| 293 | @echo '$*=$($*)' |
| 294 | @echo ' origin = $(origin $*)' |
| 295 | @echo ' flavor = $(flavor $*)' |
| 296 | @echo ' value = $(value $*)' |
Johnny Chen | 8b2c321 | 2011-01-28 17:22:29 +0000 | [diff] [blame] | 297 | |
| 298 | |
| 299 | ### Local Variables: ### |
| 300 | ### mode:makefile ### |
| 301 | ### End: ### |