New talk
This commit is contained in:
parent
9ef0c86af6
commit
0e0df26017
2
makefiles-are-cool/Makefile
Normal file
2
makefiles-are-cool/Makefile
Normal file
@ -0,0 +1,2 @@
|
||||
sorted_names.txt: names.txt
|
||||
sort $< >$@
|
BIN
makefiles-are-cool/bigbrain.png
Normal file
BIN
makefiles-are-cool/bigbrain.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 134 KiB |
BIN
makefiles-are-cool/bosch.jpg
Normal file
BIN
makefiles-are-cool/bosch.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 248 KiB |
BIN
makefiles-are-cool/brain.png
Normal file
BIN
makefiles-are-cool/brain.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 171 KiB |
BIN
makefiles-are-cool/galaxybrain.png
Normal file
BIN
makefiles-are-cool/galaxybrain.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 186 KiB |
259
makefiles-are-cool/makefiles.md
Executable file
259
makefiles-are-cool/makefiles.md
Executable file
@ -0,0 +1,259 @@
|
||||
#!/bin/env slides
|
||||
|
||||
# Makefiles Are Cool
|
||||
|
||||
_By Bruce Hill_
|
||||
|
||||
|
||||
--------------------------
|
||||
|
||||
# First Experiences with Makefiles Suck
|
||||
|
||||

|
||||
|
||||
- Some college course gave you this thing that magically
|
||||
makes your code compile.
|
||||
- Some open source project has an autogenerate script that
|
||||
makes a 20,000 line makefile.
|
||||
- Configuration options are often poorly documented.
|
||||
- Just run `make` and pray it works.
|
||||
|
||||
--------------------------
|
||||
|
||||
# Actually Make Is Good
|
||||
|
||||
- Really useful for small projects
|
||||
- Not just compiling C code
|
||||
- Anything where files are used to make other files
|
||||
|
||||
--------------------------
|
||||
|
||||
# What Even Is Make?
|
||||
|
||||

|
||||
|
||||
`make` is a command you run to compile stuff.
|
||||
|
||||
--------------------------
|
||||
|
||||
# What Even Is Make?
|
||||
|
||||

|
||||
|
||||
`make` is a bunch of similar programs (most commonly GNU)
|
||||
that look at files and determines which files need to be
|
||||
rebuilt and how.
|
||||
|
||||
--------------------------
|
||||
|
||||
# What Even Is Make?
|
||||
|
||||

|
||||
|
||||
`make` is a domain specific programming language
|
||||
and interpreter for expressing dependency trees
|
||||
and production rule systems.
|
||||
|
||||
--------------------------
|
||||
|
||||
# What Even Is Make?
|
||||
|
||||

|
||||
|
||||
`make` is a lightweight way to have files turn
|
||||
into other files lazily using simple rules.
|
||||
|
||||
--------------------------
|
||||
|
||||
# Makefiles
|
||||
|
||||
When you run `make`, it looks for a file called
|
||||
`Makefile` or `makefile` with a set of rules.
|
||||
|
||||
Each rule has the format:
|
||||
|
||||
```make
|
||||
output: input1 input2 input3...
|
||||
how to build...
|
||||
```
|
||||
|
||||
The body of the rule is a _shell script_
|
||||
|
||||
**The rule will only run if any of the input files has
|
||||
been modified more recently than the output file!**
|
||||
|
||||
----------------------------
|
||||
|
||||
# Simple Example
|
||||
|
||||
```make
|
||||
sorted_names.txt: names.txt
|
||||
sort names.txt > sorted_names.txt
|
||||
```
|
||||
|
||||
If you ever modify `names.txt`, you can run
|
||||
`make sorted_names.txt` to create the sorted names file.
|
||||
|
||||
If you don't want to retype the filename, you can use
|
||||
the variables `$<` (input files), `$@` (output file),
|
||||
and `$^` (first input file):
|
||||
|
||||
```make
|
||||
sorted_names.txt: names.txt
|
||||
sort $< > $@
|
||||
```
|
||||
|
||||
-----------------------------
|
||||
|
||||
# Another Example
|
||||
|
||||
Run a python script to draw an image:
|
||||
|
||||
```make
|
||||
graph.png: plot_graph.py graph_module.py
|
||||
python3 $^
|
||||
```
|
||||
|
||||
This will re-run if either `plot_graph.py` or
|
||||
`graph_module.py` is edited.
|
||||
|
||||
----------------------------
|
||||
|
||||
# Dependency Trees
|
||||
|
||||
Rules can form dependency trees:
|
||||
|
||||
```make
|
||||
graph.png: plot_graph.py data.csv
|
||||
python3 plot_graph.py data.csv
|
||||
|
||||
data.csv: raw_data.bin
|
||||
convert_bin_to_csv raw_data.bin
|
||||
```
|
||||
|
||||
Running `make graph.png` will first make sure that
|
||||
`plot_graph.py` and `data.csv` are up-to-date,
|
||||
then make sure that `graph.png` is up-to-date.
|
||||
|
||||
Rules are **lazy**, so if you edit `plot_graph.py`, but
|
||||
don't touch `raw_data.bin`, then running `make graph.png`
|
||||
won't run `convert_bin_to_csv` because it's not necessary.
|
||||
|
||||
----------------------------
|
||||
|
||||
# Phony Rules
|
||||
|
||||
Sometimes you have a rule that doesn't actually have
|
||||
a file ouput.
|
||||
|
||||
You can use `.PHONY` to tell `make` to not bother looking
|
||||
for a file with that name.
|
||||
|
||||
Most commonly used for `all`, `clean`, and `install`:
|
||||
|
||||
```make
|
||||
.PHONY all clean install
|
||||
|
||||
all: sorted_names.txt graph.png
|
||||
|
||||
clean:
|
||||
rm sorted_names.txt graph.png
|
||||
|
||||
install: sorted_names.txt graph.png
|
||||
cp $< /etc/cool-files/
|
||||
```
|
||||
|
||||
**Note 1:** a rule with no body just means "I need these
|
||||
files to be built using their rules, but I don't do anything
|
||||
in particular"
|
||||
|
||||
**Note 2:** when you run just `make` with no arguments, it
|
||||
runs the _first_ rule in the file, which is usually `all`
|
||||
|
||||
----------------------------
|
||||
|
||||
# Pattern Rules
|
||||
|
||||
If you put a `%` in the name, you can do reusable pattern
|
||||
rules that work for different filenames.
|
||||
|
||||
## Turn Markdown Files into PDFs
|
||||
|
||||
```make
|
||||
%.pdf: %.md
|
||||
pandoc $< -o $@
|
||||
```
|
||||
|
||||
## Compile C Code
|
||||
|
||||
```make
|
||||
%.o: %.c %.h
|
||||
cc -c $^ -o $@
|
||||
```
|
||||
|
||||
-----------------------------
|
||||
|
||||
# Defining Variables
|
||||
|
||||
You can define variables that are reused in different
|
||||
rules like this:
|
||||
|
||||
```make
|
||||
O=-O2
|
||||
CFLAGS=-Wall -Werror $O
|
||||
|
||||
all: program
|
||||
|
||||
program: main.c foo.o baz.o
|
||||
cc $CFLAGS $< -o $@
|
||||
|
||||
%.o: %.c %.h
|
||||
cc $CFLAGS -c $^ -o $@
|
||||
```
|
||||
|
||||
When running `make` you can override these variables if you
|
||||
want to:
|
||||
|
||||
```bash
|
||||
make O=-O1 CFLAGS='-Weverything -Werror' all
|
||||
```
|
||||
|
||||
**Warning:** changing the variables does not change file
|
||||
modification times, so `make` won't know stuff needs to
|
||||
be rebuilt with different flags.
|
||||
|
||||
--------------------------------
|
||||
|
||||
# My Website
|
||||
|
||||
I use a makefile to build my website and blog:
|
||||
|
||||
- `pandoc` to make HTML files from markdown files.
|
||||
- A janky bash script to make an RSS feed from my blogposts.
|
||||
- `gzip` to compress my CSS files.
|
||||
- `rsync` PHONY rule to send files to my server.
|
||||
|
||||

|
||||
|
||||
--------------------------------
|
||||
|
||||
# Not Covered
|
||||
|
||||
There's a bunch of stuff not covered in the talk:
|
||||
|
||||
- Platform-dependent variables
|
||||
- Pattern matching variables (get all the files of a type)
|
||||
- Built-in rules (you can run `make` even without a makefile)
|
||||
- Silent commands with `@`
|
||||
- [Read the docs](https://www.gnu.org/software/make/manual/html_node/index.html)
|
||||
- You can use pandoc to make a website
|
||||
- You can use pandoc to make manpages for programs
|
||||
- You can use pandoc for anything
|
||||
- Please don't make scripts to autogenerate 10K+ line makefiles
|
||||
|
||||
-------------------------------
|
||||
|
||||
# The End
|
||||
|
||||
Thanks! Hope you try using simple
|
||||
Makefiles for your own projects!
|
3
makefiles-are-cool/names.txt
Normal file
3
makefiles-are-cool/names.txt
Normal file
@ -0,0 +1,3 @@
|
||||
Carol
|
||||
Alice
|
||||
Bob
|
3
makefiles-are-cool/sorted_names.txt
Normal file
3
makefiles-are-cool/sorted_names.txt
Normal file
@ -0,0 +1,3 @@
|
||||
Alice
|
||||
Bob
|
||||
Carol
|
BIN
makefiles-are-cool/tinybrain.png
Normal file
BIN
makefiles-are-cool/tinybrain.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 120 KiB |
43
makefiles-are-cool/website_makefile.make
Normal file
43
makefiles-are-cool/website_makefile.make
Normal file
@ -0,0 +1,43 @@
|
||||
POSTS := $(shell find posts/ -name '*.md')
|
||||
POST_SYMLINKS = $(POSTS:posts/%.md=%.html)
|
||||
DRAFTS := $(shell find drafts/ -name '*.md')
|
||||
MD_FILES := $(shell find . -path ./media -prune -false -o -name '*.md')
|
||||
MD_GEN = $(MD_FILES:%.md=%.html)
|
||||
DEST=bruce@bruce-hill.com:/var/www/htdocs/blog.bruce-hill.com
|
||||
|
||||
all: $(MD_GEN) index.html drafts/index.html archive.html rss.xml symlinks style.gz.css
|
||||
|
||||
symlinks: $(POST_SYMLINKS)
|
||||
|
||||
$(POST_SYMLINKS):
|
||||
ln -sf "posts/$@" "$@"
|
||||
|
||||
clean:
|
||||
rm -f $(MD_GEN) posts/*.html posts/*/index.html drafts/*.html drafts/*/index.html index.html archive.html rss.xml
|
||||
|
||||
index.html: .staticgen/index.sh $(POSTS) .pandoc/defaults.yml .pandoc/templates/default.html .pandoc/templates/blogpreview.html
|
||||
.staticgen/index.sh >$@
|
||||
|
||||
archive.html: .staticgen/archive.sh $(POSTS) .pandoc/defaults.yml .pandoc/templates/blogarchive.html
|
||||
.staticgen/archive.sh >$@
|
||||
|
||||
rss.xml: .staticgen/rss.sh $(POSTS) .pandoc/defaults.yml .pandoc/templates/rss.xml
|
||||
.staticgen/rss.sh >$@
|
||||
|
||||
drafts/index.html: .staticgen/drafts-index.sh $(DRAFTS) .pandoc/defaults.yml .pandoc/templates/blogpreview.html
|
||||
.staticgen/drafts-index.sh >$@
|
||||
|
||||
posts/%.html: posts/%.md .pandoc/defaults.yml .pandoc/templates/default.html
|
||||
pandoc --defaults .pandoc/defaults.yml -L .pandoc/prev-next.lua --metadata=url:/$(subst posts/,,$(subst .html,,$@)) $< -o $@
|
||||
|
||||
%.html: %.md .pandoc/defaults.yml .pandoc/templates/default.html
|
||||
pandoc --defaults .pandoc/defaults.yml --metadata=url:/$(subst .html,,$@) $< -o $@
|
||||
|
||||
%.gz.css: %.css
|
||||
gzip -ck $< >$@
|
||||
|
||||
sync: all
|
||||
rsync -PuzzlOJtrh --delete --chmod=g+w --chown=:www-data --exclude-from=.rsync-exclude ./ $(DEST)
|
||||
|
||||
.PHONY: all sync clean symlinks
|
||||
# vim: ft=make
|
Loading…
Reference in New Issue
Block a user