Compile your programs faster with Makefile.

A Necessary Contrast

Before we talk about Makefiles, it's important to understand the difference between compiled and interpreted languages:

  • Interpreted languages (like Python or JavaScript) execute code line by line, without requiring a prior compilation step. You just write the code and run it directly with the interpreter.

  • Compiled languages (like C or C++) require an intermediate step: converting the source code into an executable file (machine language or binary) using a compiler.

And this is where the problem arises. In small projects, compiling a single file is easy, but in larger projects, doing it manually becomes unmanageable.

From personal experience, when I started developing in C and working on practice projects, compilation wasn't a problem. An example would be this basic program that shows the classic Hello, World! message in the terminal:

#include <stdio.h>

int main(void)
{
    printf("Hello World!\n");
    return (0);
}

To compile it, all we would need to type in the terminal is:

cc main.c -o hello_world // <-- Compile
./hello_world // <-- Run the program
Hello World! // <-- Output

But what if we have multiple files in our project? Imagine we have the following structure:

proyect/
│── main.c
│── stack.c
│── stack.h

If main.c uses functions in stack.c, we should compile it like this:

cc -c stack.c  # Make stack.o
cc -c main.c  # Make main.o
cc main.o stack.o -o program

This compilation process becomes tedious and error-prone in large projects. That's where Makefile comes in.

What is a Makefile?

A Makefile is a file where we can define certain rules to compile a program efficiently using the make command.

A basic Makefile for our previous example might look like this:

main: main.c stack.o
    cc -Wall -g -o main main.c stack.o 

stack.o: stack.c stack.h
    cc -Wall -c stack.c

But let's understand what each part of this does.

1. The Target

main: main.c stack.o

Here, we define a target, in this case, main. When we run make main, make will look for a target with that name and check its dependencies.

  • main: β†’ Defines the target we want to build.

  • main.c stack.o β†’ These are the files necessary to build main (the dependencies).

πŸ’‘ This means that make will only build main if these files exist. If one is missing, it will try to generate it automatically.

2. The Command

This is the command that will run if we need to build main:

cc -Wall -g -o main main.c stack.o
  • cc β†’ The compiler we're using (could be cc).

  • -Wall β†’ Enables all common warnings.

  • -g β†’ Includes debugging information.

  • -o main β†’ Specifies the name of the generated executable.

  • main.c stack.o β†’ These are the files being compiled to create main.

πŸ’‘ Important note: lines with commands must be indented with a tab, not spaces. If you use spaces, make will throw an error.

3. The Rule

stack.o: stack.c stack.h
    cc -Wall -c stack.c

Here, we tell make how to build stack.o.

  • stack.o: β†’ The target we want to generate.

  • stack.c stack.h β†’ These are the dependencies needed to create it.

If stack.o doesn't exist or if stack.c has changed since the last compilation, make will run this command:

cc -Wall -c stack.c
  • -c β†’ Indicates we want to compile stack.c without generating an executable, just an intermediate object file (stack.o).

πŸ’‘ This means that if stack.c changes, make will recompile only that file instead of recompiling the entire project, saving time.

What Happens When We Run make main?

  1. make checks if main exists and if it’s up to date.

  2. If main doesn’t exist or any of its files have changed, it checks main.c and stack.o.

  3. If stack.o doesn't exist, it follows its rule and compiles stack.c into stack.o.

  4. Finally, it compiles main.c along with stack.o to generate main.

πŸ—‘ Rules for Deleting Files in Makefile

So far, we've seen that compilation generates many intermediate files (.o files and executables). These files can clutter up our workspace and should be deleted before sharing the code. As we’ve seen, deleting them manually every time we compile isn’t ideal.

Instead of doing it manually, we can define rules in the Makefile to do it automatically.

πŸ“Œ 1. Defining the clean Rule

.PHONY: clean
clean: 
    rm -rf *.o main

πŸ’‘ What does this do?

  1. .PHONY: clean

    • Indicates that clean is a command, not a file, to avoid confusion with any files that may have the same name.
  2. clean:

    • Defines the cleaning task.
  3. rm -rf *.o main

    • rm β†’ The Linux command to remove files.

    • -r β†’ Removes directories recursively.

    • -f β†’ Forces removal without confirmation.

    • *.o β†’ Removes all object files (.o).

    • main β†’ Removes the executable main.

Now, when we run:

make clean

It will delete all compilation files, leaving only the source code.

Now that we know how to generate and delete all our files, we can improve our Makefile further by using variables. Let’s see how.

βš™οΈ Using Variables in Makefile

We can see that certain elements repeat often, like the compiler cc or the compilation flags -Wall -g. To avoid repeatedly writing this and make the Makefile cleaner and easier to modify, we can use variables.

Defining variables is easy. Normally, they’re placed at the beginning of the file, before defining the compilation rules.

# Define variables
CC     = cc
CFLAGS = -Wall -g
OBJS   = main.o stack.o 

main: main.c stack.o
    $(CC) $(CFLAGS) -o main main.c stack.o 

stack.o: stack.c stack.h
    $(CC) $(CFLAGS) -c stack.c

.PHONY: clean
clean:
    rm -rf *.o

Now, instead of writing the compiler directly, we use the variable. So, if for some reason we want to switch from cc to gcc, we just need to update the value of the CC variable, and this change will be reflected in all the places where we use that variable.

Now, instead of writing all the commands manually, we can simply run:

make

Make will check if there are changes in stack.c or main.c, and it will only recompile the necessary files.

Alternatively, if we want to remove all object files:

make clean

With everything we’ve learned so far, let’s see how a real-world project’s Makefile might look:

#    FLAGS _____________________________
CC            =    cc
FLAGS        =    -Wall -Werror -Wextra -g3 -fsanitize=address
RM            =    rm -f

#    CONFS _____________________________
SERVER_EXE    =    server 
CLIENT_EXE    =    client

#    SRCS ______________________________
SERVER_SRC    =    ./src/server.c    
CLIENT_SRC    =    ./src/client.c
BANNERS_SRC =    ./utils/misc/banners.c
LIBFT_DIR    =    ./utils/libft
PRINTF_DIR    =    ./utils/ft_printf
LIBFT        =    $(LIBFT_DIR)/libft.a
PRINTF        =    $(PRINTF_DIR)/libftprintf.a

#    OBJS ______________________________
SERVER_OBJ    =    $(SERVER_SRC:.c=.o)
CLIENT_OBJ    =    $(CLIENT_SRC:.c=.o)
BANNER_OBJ    =    $(BANNERS_SRC:.c=.o)

all: $(SERVER_EXE) $(CLIENT_EXE) $(BANNER_OBJ)

$(BANNER_OBJ): $(BANNERS_SRC)
    @$(CC) $(FLAGS) -c $(BANNERS_SRC) -o $(BANNER_OBJ)

$(SERVER_EXE): $(SERVER_OBJ) $(BANNER_OBJ) $(LIBFT) $(PRINTF)
    @$(CC) $(FLAGS) $(SERVER_OBJ) $(BANNER_OBJ) $(LIBFT) $(PRINTF) -o $(SERVER_EXE)
    @echo "βœ… $(SERVER_EXE) compiled success."

$(CLIENT_EXE): $(CLIENT_OBJ) $(BANNER_OBJ) $(LIBFT) $(PRINTF)
    @$(CC) $(FLAGS) $(CLIENT_OBJ) $(BANNER_OBJ) $(LIBFT) $(PRINTF) -o $(CLIENT_EXE)
    @echo "βœ… $(CLIENT_EXE) compiled success."

$(LIBFT): 
    @make -sC $(LIBFT_DIR) 

$(PRINTF):
    @make -sC $(PRINTF_DIR)

%.o: %.c
    @$(CC) $(FLAGS) -c $< -o $@ 

clean:
    @$(RM) $(SERVER_OBJ) $(CLIENT_OBJ) $(BANNER_OBJ)
    @make -sC $(LIBFT_DIR) clean
    @make -sC $(PRINTF_DIR) clean
    @echo "All objects files cleaned"

fclean: clean
    @$(RM) $(SERVER_EXE) $(CLIENT_EXE) $(LIBFT)
    @make -sC $(LIBFT_DIR) fclean
    @make -sC $(PRINTF_DIR) fclean
    @echo "All executables and libraries cleaned"

re: fclean all

.PHONY: all clean fclean re

🎯 Conclusion: Why Use Makefile?

βœ… Automates compilation and avoids manually writing long commands.
βœ… Optimizes recompilations, only recompiles what's necessary.
βœ… Eases maintenance, preventing human errors.
βœ… It's a standard in C and C++ development, used in large projects.

If you're not using Makefiles in your projects yet, now is the perfect time to start! πŸš€

πŸ“₯ Want a template to use in your projects?

Here you can πŸ”— download Makefile Template

1
Subscribe to my newsletter

Read articles from Cristian Zapata Arias directly inside your inbox. Subscribe to the newsletter, and don't miss out.

Written by

Cristian Zapata Arias
Cristian Zapata Arias

Hello there! I'm Cristian Zapata πŸ‘½ a graphic designer, visual artist and front-end developer focused on JavaScript, Angular and Typescript. Beyond the screen, I'm fascinated by the intricacies of systems, networks, and security, constantly expanding my knowledge in scripting and self-configuring Neovim setups with plugins. I'm also passionate about crafting unique keyboards and merging art with technology to create something truly special. Let's explore the endless possibilities together! πŸš€