Un aperçu de la technologie GPR
La mise en œuvre des différents éléments constitutifs d’un projet logiciel peut s’avérer être une tâche fastidieuse. C’est pour cela que, depuis de nombreuses années, un ensemble d’outils ont été développés afin de faciliter, de sécuriser et de systématiser celle-ci.
Cet article est une introduction à l’une des solutions existantes qu’est la technologie GNAT GPR (Gnat PRoject).
Cet outil permet de gérer simplement et efficacement des projets multi-langages de tailles et de natures extrêmement variables. Même si cette technologie a été introduite dans le cadre de la mise en œuvre de projets Ada, elle n’est pas dévolue à ce langage.
Généralités¶
La technologie GPR permet de construire et de gérer simplement et efficacement des projets multi-langages.
Les fichiers GPR peuvent dépendre d’autres fichiers GPR de façon modulaire.
Cette modularité simplifie l’intégration de systèmes complexes tout en permettant la réutilisation de projets existants.
La technologie s’appuie entre autres sur l’outil GPRbuild
.
Celui-ci interprète un fichier GPR (.gpr
) contenant une description du projet.
Il y a plusieurs types de GPR et donc de projets :
les projets « normaux »,
les projets agrégeant d’autres projets « agregate »,
les projets étendant d’autres projets « extended »,
les projets librairie « library »,
les projets abstraits « abstract ».
À ces types viennent se greffer des notions de « limited » et de projets enfants. Les projets limités permettent de gérer des dépendances circulaires entre projets. Les projets enfants reprenant un peu la notion d’unité enfant Ada transposée à la notion de projet.
Un fichier GPR décrit entre autres :
les fichiers sources contenant les points d’entrée,
des définitions de types et de variables propres au projet,
le ou les langages utilisés,
les répertoires contenant les fichiers sources,
le répertoire dans lequel les produits de compilations doivent être placés (
*.ali
,.o
, etc.),le répertoire contenant les binaires ou les librairies générés,
les paramètres à appliquer aux outils utilisés (compilateur, binder, éditeur de lien, dévermineur, etc.),
…
Cette description se fait au travers d’une syntaxe proche de l’Ada.
with "../configuration.gpr"; library project Convol is for Object_Dir use "_obj"; for Library_Name use "convol"; for Library_Dir use "_lib"; for Source_Dir use ("."); for Language use ("C"); package Builder renames Configuration.Builder; package Compiler renames Configuration.Compiler; end Convol;
Dans le cadre de projets plus ou moins complexes, l’articulation des GPR reflète l’architecture de ou des applications et éventuellement celle du système.
Note
En conséquence, une articulation compliquée et peu claire des GPR est probablement le reflet d’un problème bien plus profond à savoir une mauvaise architecture/découpe dans son ensemble de l’application ou du système.
Un browser intégré à l’environnement de développement GPS (GNAT Programming Studio) permet de visualiser clairement les dépendances entre GPR. Ci-dessous un projet type mis en œuvre par Systerel qui est complexe mais qui reflète aussi une bonne structuration.
Exemple¶
Cet exemple très simple va présenter les GPR de types :
projet « normal »,
projet librairie « library »,
projet abstrait « abstract ».
Ainsi que la mise en œuvre :
des préprocesseurs Ada du GNAT et C,
d’un projet multi-langages.
Dans cet exemple on décrit au travers des GPR :
le fichier source contenant le point d’entrée,
la définition d’un type de build et d’une variable de ce même type,
les langages utilisés,
le répertoire dans lequel les produits de compilations doivent être Placés,
le répertoire contenant le binaire et la librairie générés,
les paramètres à appliquer aux compilateurs.
L’application générée réalise la convolution d’une image. L’ossature est en Ada 2012 et le code implémentant le produit de convolution est en C. On veut pouvoir avoir un build DEBUG et un build RELEASE de l’application. Ces deux types de build sont différents aussi bien sur les paramètres de génération que sur le comportement du programme qu’ils produisent.
L’arborescence projet est la suivante :
Avec :
config.gpr : GPR « abstrait » permettant de mettre en commun un certain nombre de définitions,
test_convol.gpr : GPR « normal » générant l’exécutable « test_convol.exe »,
convol.gpr : GPR « librairie » générant la librairie statique « libconvol.a »,
test_convol.adb : point d’entrée Ada de l’application,
image.ad[sb] : paquetage définissant une image et implémentant ses services,
convol.c : implémentation du produit de convolution.
Note
Pour ce qui suit et dans des cas simples le pragma Debug
peut aussi
satisfaire
au même besoin en ce qui concerne le comportement du programme suivant
le type de build 2.
Il est assez courant de définir un build DEBUG
et RELEASE
ainsi
qu’une target NATIVE
et CROSS
.
Les codes mis en œuvre sont succinctement :
config.gpr¶
Par le biais de ce GPR « abstrait », on va à la fois définir le type de build et les définitions partagées entre GPR.
Plus précisément, on va définir ce qu’est un build DEBUG
et un build
RELEASE
ainsi que les options de compilation associées.
Pour le build DEBUG
, on positionne les macros __DEBUG__
et
Build_Kind
qui seront pris en compte lors de la phase de
preprocessing des sources.
Pour le build RELEASE
, on considère les warnings des compilateurs
comme des erreurs.
Au final le fichier config.gpr sera :
convol.gpr¶
Par le biais de ce GPR « librairie », on va fournir les
informations requises pour la construction de la librairie statique
libconvol.a
. On note aussi que la configuration du compilateur est
celle définie par le GPR config.
test_convol.gpr¶
Par le biais de ce GPR « normal », on va expliciter le point d’entrée Ada de notre programme. Tout comme le précédent GPR, il s’appuie sur la configuration du compilateur définie par le GPR config.
Pour générer l’application on peut soit utiliser l’IDE GPS et sélectionner le build souhaité dans la vue scenario.
Soit appeler directement l’outil GPRbuild
en ligne de
commande en spécifiant le build souhaité.
Dans les deux cas, les produits de compilation seront les mêmes :
Conclusion¶
La technologie GNAT GPR apporte une solution efficace et évolutive à la mise en place de projets pouvant être à la fois volumineux et complexes. Cette technologie s’intègre très aisément au sein d’un environnement de développement.