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