Shell linux, creation des makefiles, commande make

La commande MAKE

Introduction

Ce tutoriel s'adresse aux débutants programmeurs sous Linux. Il vous présente la commande make et la création de fichier « makefile ». Je précise sous Linux, mais sachez qu'ils existent aussi sous Windows, vous pouvez demander à VC++ de vous les construire.

Alors qu'est ce qu'un fichier makefile?
C'est un fichier nommé « makefile » qui permet, en ne tapant que « make » , dans le répertoire où il se trouve de compiler tout votre projet.

Par conséquent il rassemble toute les directives de compilation de votre projet.

Vous imaginez bien qu'à partir d'un certain nombre de fichier source, il devient tout à fait indispensable.

Imaginez-vous taper gcc fichier.c -Wall -c etc... pour chacun de vos fichiers!!! Make fait tout :)

L'édition d'un fichier Makefile

Un fichier « makefile » s'édite à l'aide d'un éditeur de texte. Je vous conseille personnellement Xemacs.

Les Commentaires sont unilignes et commencent par #:

 
# commentaire...

Un tel fichier contient un ensemble de règle, un règle se compose de la manière suivante:

cible : dépendances
<tab>   commandes

Remarque : <tab> correspond à une véritable tabulation..

La cible est le fichier à construire, les dépendances sont les fichiers dont on a besoin pour créer la cible. Les commandes sont les commandes à exécuter pour générer la cible.

Prenons un exemple, pour créer un fichier *.o nous avons besoin d'un fichier *.c et de fichiers *.h

main.o : main.c main.h autre.h
<tab>   gcc main.c -c -Wall

Ici si le fichier make ne contient que ça, le lancement de la commande make va générer main.o.

L'intérêt de make réside dans le fait, qu'il ne reconstruit une cible que si les dépendances ont et modifier depuis la dernière utilisation de make, ou si la cible n'existe pas. Ce qui évite de tout recompiler à chaque fois.

Pour construire une cible, make se relance récursivement. Ce qui fait que l'ordre des règles importe peut.

Les macros

Ce sont des variables de type chaîne de caractères.
Définition: MACRO = Valeur
Référence: $(MACRO) -> sert à son utilisation

Exemple:

MODULE = main.o aux.o

main : $(MODULE)
<tab>   gcc $(MODULE) -o main

Il en existe des prédéfinis: les macros internes

  • $@ Contient le nom de la cible
  • $* Contient de nom de la cible sans suffixe ( si cible = main.c; $* = main)
  • $^ Contient le nom de toutes les dépendances
  • $? Liste des dépendances dont la date est postérieure à la cible
  • $< Contient la dépendance qui a provoqué la reconstruction de la cible

Les macros prédéfinies que l'on peut modifier:

  • $(CC) Contient le nom du compilateur
  • $(CPPFLAGS) Liste des options de pré-processage
  • $(CFLAGS) Liste des options de compilation
  • $(LDFLAGS) Liste des options de pré-édition des liens.

Un petit exemple pour illustrer tout ça :)

CC = gcc
CFLAGS = -Wall -O3
LDFLAGS = -lm 

programme : main.o matrice.o
<tab>   $(CC) $^ -o $@ $(LDFLAGS) 

main.o  : main.c matrice.h
<tab>   $(CC) main.c -c $(CFLAGS)

matrice.o : matrice.h matrice.c
<tab>   $(CC) $*.c -c $(CFLAGS)

Ici, l'exécution de la commande make, compilera le fichier 'programme ', en générant les fichiers *.o intermédiaires.

Remarque: le lancement de make ne génère que la première cible, et génère les autres si ils en a besoin pour la construire.

Pour compiler une cible particulière, on lance make avec le nom de la cible:

$ make matrice.o         // Genère le fichier matrice.o

Les directives

Ce sont des options pour le make. Elles sont à placer dans le fichier makefile.
En voici quelques unes:

  • .INIT : <commande à exécuter au début du make>
  • .DEFAULT : <commande à exécuter en cas d'erreur >
  • .DONE : <commande exécuter à la fin de make>
  • .IGNORE: Evite au make de s'arrêter en cas d'erreur
  • .SILENT: N'affiche pas les commandes exécutées
  • .PRECIOUS: <fichier>* Ne pas détruit pas les fichiers ..
  • .PHONY : <cible>* Déclares les cibles qui ne sont pas des fichiers

Exemple:
On veut créer une cible clean qui ne correspond pas à un fichier, qui une fois appeler dans le make va nettoyer le dossier actuel. On déclare donc clean comme cible non fichier.

.PHONY: clean           

clean:
<tab>   rm -f *.o
<tab>   rm -f *~

Ainsi la commande $ make clean , va supprimer tous les fichiers *.o et les fichier *~

En général on définie également une cible « all » qui regroupe toutes les fichiers à produire.

De facon à ce que la commande $ make all construise tous les fichiers désirés.

.PHONY: clean all

all: prog1 prog2 prog3 

Ainsi, la commande $make all ordonne la compilation des 3 programmes.

Il reste encore beaucoup de chose à dire sur le makefile, mais je vous laisse en regarder là doc, vous savez à présent le principal :).

Exemple complet

Je termine ainsi par un exemple complet d'un projet devant générer deux programmes:

# Début du make file
# Déclaration des macros

CC = gcc
CFLAGS = -Wall -O3
LDFLAGS = -lm 

.PHONY:  all clean           

all : prog1 prog2 

prog1 : main.o affichage.o lecture.o
<tab>    $(CC) $^ -o $@ $(LDFLAGS)

prog2 : main2.o affichage.o matrice.o
<tab>    $(CC) $^ -o $@ $(LDFLAGS)

main2.o : main2.c affichage.h matrice.h
<tab>    $(CC) main2.c -c $(CFLAGS)

main.o : main.c affichage.h lecture.h
<tab>    $(CC) main.c -c $(CFLAGS) 

affichage.o : affichage.h affichage.c
<tab>    $(CC) affichage.c -c $(CFLAGS)

# etc...

 clean:
<tab>    rm -f *.o
<tab>    rm -f *~  
 

La commande $make all, générera ainsi prog1 et prog2.
La commande $make prog2, ne générera que le prog2.

Ce document intitulé « Shell linux, creation des makefiles, commande make » issu de CodeS SourceS (codes-sources.commentcamarche.net) est mis à disposition sous les termes de la licence Creative Commons. Vous pouvez copier, modifier des copies de cette page, dans les conditions fixées par la licence, tant que cette note apparaît clairement.
Rejoignez-nous