Prev | Contents | Next |
Knowing what grit is capable of is only the first step. To use it comfortably, you need to know how to integrate it into the standard build process of your project. In this chapter, I'll explain how to do that. First up is, using grit with or without .grit configuration files. After that, I'll show how to use these rules on the standard template makefiles. Finally, I'll describe how to split off the image conversion to a separate process, keeping the interference to the main makefile to a minimum. Part of this is step is also putting the graphics into an archive (.a) and creating a master header file.
The basic format of a grit command is as follows
grit source files options
The list of options start at the first argument that begins with a hyphen, '-'. You have have one or more source files, but you cannot use wildcards. The output file can be specified with the -o flag, but it is not required. If no options are given, grit will try to guess what you meant from the file itself, but this can be a bit dangerous. It is advised to give at least graphics type (-gb/-gt for bitmaps/tiles), bitdepth (-gB num) and probably output filetype as well (-ft type). Examples:
# Single bitmap # Input: foo.bmp # Output: foo.s, 16bpp bitmap as asm array named fooBitmap grit foo.bmp -fts -gb -gB16 # Sprite sheets (32x32p sprites) # Input mario.png, luigi.png # Output: karts.s, 4bpp tiles in 4x4t groups as arrays marioTiles and # luigiTiles, no palette. grit mario.png luigi.png -o karts.s -gt -gB4 -Mw4 -Mh4 -p!
The previous examples would work if you'd enter them on the command-line manually, or via a batchfile. for makefiles, you'd need a little bit extra. The (near) equivalent of the previous examples in makefilese are:
GRIT := grit.exe # foo.bmp -> foo.s ; fooBitmap = 16bpp bitmap. foo.s foo.h : foo.bmp $(GRIT) $< -fts -gb -gB16 # mario.png, luigi.png -> karts.s ; marioTiles, luigiTiles as 4bpp tiles. karts.s karts.h : mario.png luigi.png $(GRIT) $^ -o$* -fts -gt -gB4 -Mw4 -Mh4 -p!
The terms $<
, $^
and $*
are
some of makefile's automatic variables; they stand for the first
prerequisite, all prerequisites and the 'stem' of the targets,
respectively. In these rules, that equates to
$<
= foo.bmp
,
$^
= mario.png luigi.png
and
$*
= karts
.Note that in the second
rule, -o$*
doesn't add an extension. This is alright
because -fts
takes care of that.
I should point out that
the use of $*
is somewhat iffy. Sometimes it works just
fine, but in other cases it's empty, which mucks up everything. Usually,
$@
works just as well.
Having separate rules for each image doesn't exactly scale well. For this reason, grit also allows you to read the flags from configuration files, usually with the .grit extension. With these files, you can put the image-specific options inside the config file and call them with a generalized rule.
You can indicate the config file with the -ff option. For
example, `-ff foo.grit' would read additional options from
foo.grit. The config file uses the '#
' character
for comments, much like makefiles. The config file for foo.bmp
would look something like this:
# # foo.grit : example of a .grit file # -gb # Bitmapped output -gB16 # 16 bpp
The rule to go with this would be something like the following. Here you have a pattern rule that likes a .s and .h file to a .png and .grit file, so that if either the png or grit file is altered, the rule is run again. Technically, there should be a `-ff $*.grit' argument here as well, but grit always looks for an appropriate grit-file automatically, so I can omit it here.
Note that the -fts option is still in the rule itself. The extra options from the .grit-file are simply added to the set of command-line options; they do not replace them.
# Main rule.
%.s %.h : %.png %.grit
$(GRIT) $< -fts
This procedure works out nice if you have lots of different options for different files, but when you have a great many images that use the same conversion, you really don't want to have to write grit-files for each image. It would be preferable to have a single grit-file that can be used by all.
In most cases, you'd have to write explicit rules for this because you can't get the name of the grit-file from the names of the images. There is one exception to this, namely when you have a whole directory of images. If you then base the name of the grit-file on the directory name (say, dir/dir.grit), you can use this:
# Using dir.grit for a whole directory.
%.s %.h : %.png
$(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit
The tricky part is `$(<D)/$(notdir $(<D)).grit'. This takes the directory part of a source image and turns it into a full path for the grit-file. For example, for /c/stuff/dir/foo.png, it gives /c/stuff/dir/dir.grit.
The single-file rule (`%.png %.grit') and the directory rule (`%.png') can work in the same makefile, but you do have to be little careful with it. When make checks for matching files to rules, it'll use the first rule that gives a match. You need to have the single-file rule before the directory-rule. In that case the single-file rule is used if a png-file does have an accompanying grit-file, but it'll fall back on the directory-rule if it's absent. If you reversed the order of the rules, it'd always use the directory-rule, which is probably not what you'd want.
These rules are for .png files. You can create similar rules of other image types, obviously.
In a makefile, if a prerequisite or combination of prerequisites follows more than one rule, the first rule that fits is used. So order the grit rules like this:
# With matching grit-file %.s %.h : %.png %.grit $(GRIT) $< -fts # No grit-file: try using dir.grit %.s %.h : %.png $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit
The standard practice for building gba/nds homebrew apps nowadays is to use the template makefiles that come with the devkitPro distribution, or derivatives thereof. These makefiles may have to be altered in several places to make them work for grit. Here is a summary of you may need to do.
The GRAPHICS variable is for graphics what SOURCES is for the source files: it's a list of all the directories with graphics. Technically, creating a GRIT variables isn't required, but it can be useful if you have different grit versions around. Add GRIT somewhere at the top, near the inclusion of gba_rules or nds_rules, and GRAPHICS near SOURCES.
include $(DEVKITARM)/ds_rules
GRIT := grit
# ... stuff ...
TARGET := $(notdir $(CURDIR))
BUILD := build
SOURCES := source
DATA := data
INCLUDES := include
GRAPHICS := gfx
The graphics directory also needs to be added to VPATH:
export VPATH := $(foreach dir, $(SOURCES), $(CURDIR)/$(dir)) \ $(foreach dir, $(DATA), $(CURDIR)/$(dir)) \ $(foreach dir, $(GRAPHICS), $(CURDIR)/$(dir))
Next, we need a list of all the graphics files. This is the analogon of CFILES for the C sources, and it's placed in the same spot.
CFILES := $(foreach dir, $(SOURCES), $(notdir $(wildcard $(dir)/*.c))) CPPFILES := $(foreach dir, $(SOURCES), $(notdir $(wildcard $(dir)/*.cpp))) SFILES := $(foreach dir, $(SOURCES), $(notdir $(wildcard $(dir)/*.s))) BINFILES := $(foreach dir, $(DATA), $(notdir $(wildcard $(dir)/*.*))) GFXFILES := $(foreach dir, $(GRAPHICS), $(notdir $(wildcard $(dir)/*.png)))
If you want BMPs as well, you can add them here in the same way, only
with +=
instead of :=
.
This is the trickiest step. OFILES lists all the object (.o) files that the build creates, and it will run along these files in the order they're listed. For this reason, the object files of the GFXFILES must occur before those of the cpp/c/s files. If not, make will try to compile/assemble the files that grit creates before they're actually created. yet.
## NOTE: add the GFXFILES line before the sources!!! export OFILES := $(addsuffix .o,$(BINFILES)) \ $(GFXFILES:.png=.o) \ $(CPPFILES:.cpp=.o) $(CFILES:.c=.o) $(SFILES:.s=.o)
Finally, add the rules for grit conversion that I gave before. The best place to add these are at the bottom of the makefile; somewhere after the $(OUTPUT) lines, but before the dependencies.
# With matching grit-file %.s %.h : %.png %.grit $(GRIT) $< -fts # No grit-file: try using dir.grit %.s %.h : %.png $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit -include $(DEPENDS) endif
For other image filetypes, add similar rules with those extensions.
Adding the rules to the main makefile works, but it has a number of disadvantages. First, of course, is that you have to change the makefile, and this can be tricky in itself for those new to makefiles. Tied to that is the fact that the order of the changes matters as well. Finally, it makes it harder to create specialized rules. The rules I created above process one image at a time, but sometimes you really need to have multiple sources, like when sharing palettes or tilesets.
In many ways, it'll be easier to handle image conversion as a completely separate process, to be done before any real source files are done. Not only are you more in control of the build order, but you can also do all the unseemly steps that you may require in image processing without it upsetting main makefile itself.
In this section, I'll illustrate how you can convert the images in different ways and archive the output into a resource library to be linked into the project like any other lib. I'll also show how you can combine the separate headers into a master header, which makes #including the declarations considerably easier. This is one step that would simply be impossible if you'd use a single makefile.
Splitting of the graphics conversion involved the use of a second makefile which I call gfxmake. Here it is in full.
# # Making a gfx library from grit output # # For use on data-arrays only. NO CODE!!! # # --------------------------------------------------------------------- # SETUP # --------------------------------------------------------------------- export PATH := $(DEVKITARM)/bin:$(PATH) GRIT := grit.exe .SUFFIXES: include $(DEVKITARM)/base_rules # --------------------------------------------------------------------- # (1) PROJECT DETAILS # --------------------------------------------------------------------- # GFXTITLE : Graphics library name # BUILD : Directory for build process temporaries. Should NOT be empty! # GFXDIRS : List of graphics directories # GFXEXTS : Graphics extensions. # General note: use . for the current dir, don't leave them empty. BUILD := build GFXDIRS := gfx level1 palmerge smkkarts GFXLIB ?= libgfx.a GFXHDR ?= all_gfx.h GFXEXTS := png bmp # --- Exceptions --- # Add files/file-variables for special rules here. Put the rules # At the bottom of the makefile. Be careful with directories, as # we'll be in $(BUILD) when converting. # GFXSPECIALS : removed from GFXFILES # OSPECIALS : added to OFILES export LOZLEVEL1 ?= $(notdir $(wildcard level1/*.png)) export PALMERGE ?= $(notdir $(wildcard palmerge/*.png)) # Key exception variables export GFXSPECIALS := $(LOZLEVEL1) $(PALMERGE) OSPECIALS := level1.o palmerge.o # --------------------------------------------------------------------- # BUILD FLAGS # --------------------------------------------------------------------- # Since there's no code to compile, we won't need optimizations, # architectures etc. CFLAGS := CXXFLAGS := $(CFLAGS) -fno-rtti -fno-exceptions ASFLAGS := # --------------------------------------------------------------------- # (2) BUILD PROCEDURE # --------------------------------------------------------------------- ifneq ($(BUILD),$(notdir $(CURDIR))) # still in main directory. export TARGET := $(CURDIR)/$(GFXLIB) export VPATH := $(foreach dir, $(GFXDIRS), $(CURDIR)/$(dir)) export DEPSDIR := $(CURDIR)/$(BUILD) GFXFILES := $(filter-out $(GFXSPECIALS), \ $(foreach dir, $(GFXDIRS), \ $(foreach ext, $(GFXEXTS), \ $(notdir $(wildcard $(dir)/*.$(ext))) \ ))) export OFILES := $(addsuffix .o, $(basename $(GFXFILES))) $(OSPECIALS) # --- More targets ---------------------------------------------------- .PHONY: $(BUILD) clean # --- Create BUILD if necessary, and run this makefile from there --- $(BUILD): @[ -d $@ ] || mkdir -p $@ @make --no-print-directory -C $(BUILD) -f $(CURDIR)/gfxmake all : $(BUILD) clean: @echo clean ... @rm -fr $(BUILD) $(TARGET) $(GFXHDR) # --------------------------------------------------------------------- else DEPENDS := $(OFILES:.o=.d) .PHONY : all all : $(TARGET) $(GFXHDR) $(TARGET) : $(OFILES) @echo Archiving into $(notdir $@) -@rm -f $@ @$(AR) rcs $@ $(OFILES) $(GFXHDR) : $(OFILES) @echo "Creating master header: $@" @$(call master-header, $@, $(notdir $(^:.o=.h)) ) # --------------------------------------------------------------------- # (3) BASE CONVERSION RULES # --------------------------------------------------------------------- # --- With separate .grit file --- %.s %.h : %.png %.grit $(GRIT) $< -fts %.s %.h : %.bmp %.grit $(GRIT) $< -fts %.s %.h : %.pcx %.grit $(GRIT) $< -fts %.s %.h : %.jpg %.grit $(GRIT) $< -fts # --- Without .grit file ; uses dirname/dirname.grit for options --- %.s %.h : %.png $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit %.s %.h : %.bmp $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit %.s %.h : %.pcx $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit %.s %.h : %.jpg $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit # --------------------------------------------------------------------- # (4) SPECIAL RULES # --------------------------------------------------------------------- level1.s level1.h : level1.grit $(LOZLEVEL1) @echo $(notdir $^) $(GRIT) $(sort $(filter %.png,$^)) -o$@ -ff $< palmerge.s palmerge.h : palmerge.grit $(PALMERGE) @echo $(notdir $^) $(GRIT) $(filter %.png,$^) -o$@ -ff $< # --------------------------------------------------------------------- # (5) UTILITY FUNCTIONS # --------------------------------------------------------------------- ## Merge all headers into a single large one for easier including. define master-header # $1 : master path, $2 separate header paths echo -e "//\n// $(notdir $(strip $1))\n//" > $1 echo -e "// One header to rule them and in the darkness bind them" >> $1 echo -e "// Date: $(shell date +'%F %X' )\n" >> $1 echo -e "#ifdef __cplusplus\nextern \"C\" {\n#endif" >> $1 cat $2 >> $1 echo -e "\n#ifdef __cplusplus\n};\n#endif\n" >> $1 endef ## if you just want to include the separate headers, use this instead of cat: # for hdr in $2 ; \ # do echo -e "#include \"$$hdr\"" >> $1 ; done; # --- odds and ends --- ## Get the title-part of filename. define title # $1: filepath $(basename $(notdir $1)) endef ## Get a valid C identifier for a name. define cident # $1: name `echo $1 | sed -e 's|^\([0-9]\)|_\1| ; s|[./\\-]|_|g'` endef ## Create a header file for a bin2s converted binary. define bin-header # $1: path, $2: identifier echo "extern const u32 $(strip $2)_size;" > $1 echo "extern const u8 $(strip $2)[];" >> $1 endef # --------------------------------------------------------------------- # DEPENDENCIES # --------------------------------------------------------------------- -include $(DEPENDS) endif
The gfxmake makefile is divided into a number of sections, of which the five numbered ones are the most important.
Under project details, you can find the most relevant items for the
graphics conversions. The BUILD
variable is the directory
that the converted items go into and GFXDIRS
are the
directories where the graphics are. The next two variables,
GFXLIB
and GFXHDR
are the names of the
graphics library and master header that gfxmake
creates.
GFXEXTS
are the extensions used for the images.
The subsection under exceptions
can be ignored for now.
The variables given there are excluded from the GFXFILES
and OFILES
, so that they can be built from custom rules.
Here the variables GFXFILES
and OFILES
are created, much like it was done in the template makefile. Note
that because we're not mixing code and data, the order of listing
in OFILES
is not important.
Note the use of GFXSPECIALS
and OSPECIALS
here. The names under GFXSPECIALS
are filtered out
of GFXFILES
because I want their conversion procedure
to follow special rules. The object files created in those rules
still have to be added to the OFILES
, of course,
and that's what OSPECIALS
is for.
Under the else
, right before the Rules section, are the
main targets. The target itself is the library file given by
GFXLIB
, which in this case is called libgfx.a.
There is also a rule to make the master header, GFXHDR
.
The latter rule uses a user defined function
master-header
, which I'll get to shortly.
These are the same rules as before, in the same order. I've already added versions for some other popular graphics types, but more could be added if required. All these rules use assenbly for their output because assembly can be converted to object files a hell of a lot faster than C sources. Since you probably never need to browse through the generated files, assembly should be the better option here.
There is an interesting point that may be worth mentioning about
how make goes about using these rules to turn graphics into
assembly into objects. The rules here go from .png in
PNGFILES
to .s, but nowhere are the .s
files mentioned explicitly. Only the o. files they become
are given (under OFILES
). Basically, the .s
files are implicit temporaries, and what make does in such a
case is to remove the temporaries at the end. Only after
running through the rules a second time will the assembly files
remain in existence.
It's a bit odd, I know but that's how it is. This can be more than a little annoying if you want to do something else to those files later, so consider yourself warned.
The normal rules will convert single images into single assembly/object files. For something like sharing tilesets among different maps, this simply won't do. For such occasions, you have to use explicit rules of some kind.
gfxmake makes this process easier through the
GFXSPECIALS
and OSPECIALS
variables. Files
added to GFXSPECIALS
won't follow the standard rules,
but can be redirected to your own custom rules. In this
particular case, I have all the rooms from the first dungeon in
Zelda 1 in the level1
directory, and I want to
convert the maps with a shared tileset. The steps of interest for
that are the following.
# Create a variable with the level1 filenames export LOZLEVEL1 ?= $(notdir $(wildcard level1/*.png)) # Add the sources to GFXSPECIALS and the output file to OSPECIALS export GFXSPECIALS := $(LOZLEVEL1) OSPECIALS := level1.o # Rule to for the conversion level1.s level1.h : level1.grit $(LOZLEVEL1) @echo $(notdir $^) $(GRIT) $(sort $(filter %.png,$^)) -o$@ -ff $<
Once GFXSPECIALS
and OSPECIALS
are set up,
you can do with those files whatever you like. The custom rule looks
similar to rule for the karts from before, except that now there is a
grit-file as well. Unfortunately, I do need some makefile trickery to
get it all done. To get the list of sources from the prerequisites, I
need to filter for .pngs only. The sort is somewhat optional,
but recommended as well. Another thing I'm doing here is to place
the grit-file first in the list. This enables me to get its name
with $<
, rather than having to filter it out as well.
make allows you to create your own functions as well.
These can help in doing special things. Though the others are useful
as well, the only one I make actual use of in this
gfxmake is master-header
:
# Merge all headers into a single large one for easier including. define master-header # $1 : master path, $2 separate header paths echo -e "//\n// $(notdir $(strip $1))\n//" > $1 echo -e "// One header to rule them and in the darkness bind them" >> $1 echo -e "// Date: $(shell date +'%F %X' )\n" >> $1 echo -e "#ifdef __cplusplus\nextern \"C\" {\n#endif" >> $1 cat $2 >> $1 echo -e "\n#ifdef __cplusplus\n};\n#endif\n" >> $1 endef # Usage example $(GFXHDR) : $(OFILES) @echo "Creating master header: $@" @$(call master-header, $@, $(notdir $(^:.o=.h)) )
The master-header
function takes two arguments: the path of the
master header that's to be created, and a list of the original headers. It
will create a single header file with a tile-stamped preface and then
copies all the separate headers into it. This means that the output of
gfxmake
is a combination of one header and one archive, which
can be put to use with more ease than a bucketful of separate files.
Of course, even with gfxmake you still need some changes to the template makefile, but they're smaller than adding all of gfxmake functionality into the template. There are two main changes to make.
The first is to add make libgfx.a
and its headers visible. This is done like adding any other library:
add -lgfx
to the LIBS
variable. Technically
you should add the appropriate paths to LIBDIRS
as well,
but since because the files are in the current dir and
BUILD
, this isn't really necessary.
The second is to invoke gfxmake from the main makefile. This comes down to adding a line that calls it to the build rule. The whole thing comes down to something like this.
## Create a gfx library variable GFXLIBS ?= libgfx.a ## Add it to the library list. LIBS := standard libs -lgfx ## Make the .elf depend on the graphics lib(s) $(OUTPUT).elf : $(OFILES) $(GFXLIBS) ## Add line to invoke gfxmake $(BUILD): @[ -d $@ ] || mkdir -p $@ @make --no-print-directory -f $(CURDIR)/gfxmake @make --no-print-directory -C $(BUILD) -f $(CURDIR)/Makefile
And that's about it. I think it should be possible to make it even easier, but this should be enough for now.
Getting grit to work inside your makefile shouldn't be too difficult if you follow these steps. The main thing to remember is to use configuration files and the following makefile rules:
# With matching grit-file %.s %.h : %.png %.grit $(GRIT) $< -fts # No grit-file: try using dir.grit %.s %.h : %.png $(GRIT) $< -fts -ff $(<D)/$(notdir $(<D)).grit
With these rules, you can convert single image either by using a similarly named grit-file, or share a grit-file for a whole directory by naming the grit-file the same basename as the directory. Adding explicit rules is also an option, of course.
You also need to add a few variables to the makefiles for bookkeeping purposes. These have to be added very carefully, because the order sometimes matters. An alternative is to pass the conversion process to another script (makefile or otherwise). This can probably save you a lot of hassle in the long run, which is why I'm using that method in grit's demo project. In the end, of course, what you use is up to you.
Prev | Contents | Next |
Grit | Wingrit |