Conception du compilateur - Guide rapide
Les ordinateurs sont un mélange équilibré de logiciels et de matériel. Le matériel n'est qu'un appareil mécanique et ses fonctions sont contrôlées par un logiciel compatible. Le matériel comprend les instructions sous forme de charge électronique, qui est le pendant du langage binaire dans la programmation logicielle. Le langage binaire n'a que deux alphabets, 0 et 1. Pour instruire, les codes matériels doivent être écrits au format binaire, qui est simplement une série de 1 et de 0. Ce serait une tâche difficile et lourde pour les programmeurs informatiques d'écrire de tels codes, c'est pourquoi nous avons des compilateurs pour écrire de tels codes.
Système de traitement du langage
Nous avons appris que tout système informatique est composé de matériel et de logiciels. Le matériel comprend un langage que les humains ne peuvent pas comprendre. Nous écrivons donc des programmes dans un langage de haut niveau, ce qui nous est plus facile à comprendre et à retenir. Ces programmes sont ensuite introduits dans une série d'outils et de composants OS pour obtenir le code souhaité pouvant être utilisé par la machine. Ceci est connu sous le nom de système de traitement du langage.
Le langage de haut niveau est converti en langage binaire en différentes phases. UNEcompilerest un programme qui convertit un langage de haut niveau en langage d'assemblage. De même, unassembler est un programme qui convertit le langage d'assemblage en langage de niveau machine.
Voyons d'abord comment un programme, utilisant le compilateur C, est exécuté sur une machine hôte.
L'utilisateur écrit un programme en langage C (langage de haut niveau).
Le compilateur C compile le programme et le traduit en programme d'assemblage (langage de bas niveau).
Un assembleur traduit ensuite le programme d'assemblage en code machine (objet).
Un outil de liaison est utilisé pour relier toutes les parties du programme ensemble pour exécution (code machine exécutable).
Un chargeur les charge tous en mémoire, puis le programme est exécuté.
Avant de plonger directement dans les concepts de compilateurs, nous devons comprendre quelques autres outils qui travaillent en étroite collaboration avec les compilateurs.
Préprocesseur
Un préprocesseur, généralement considéré comme faisant partie du compilateur, est un outil qui produit des entrées pour les compilateurs. Il traite du macro-traitement, de l'augmentation, de l'inclusion de fichiers, de l'extension de langue, etc.
Interprète
Un interpréteur, comme un compilateur, traduit un langage de haut niveau en langage machine de bas niveau. La différence réside dans la façon dont ils lisent le code source ou l'entrée. Un compilateur lit le code source entier à la fois, crée des jetons, vérifie la sémantique, génère du code intermédiaire, exécute tout le programme et peut impliquer de nombreuses passes. En revanche, un interpréteur lit une instruction à partir de l'entrée, la convertit en code intermédiaire, l'exécute, puis prend l'instruction suivante dans l'ordre. Si une erreur se produit, un interpréteur arrête l'exécution et la signale. alors qu'un compilateur lit l'ensemble du programme même s'il rencontre plusieurs erreurs.
Assembleur
Un assembleur traduit des programmes en langage assembleur en code machine. La sortie d'un assembleur est appelée un fichier objet, qui contient une combinaison d'instructions machine ainsi que les données nécessaires pour placer ces instructions en mémoire.
Éditeur de liens
Linker est un programme informatique qui relie et fusionne divers fichiers objets afin de créer un fichier exécutable. Tous ces fichiers peuvent avoir été compilés par des assembleurs distincts. La tâche principale d'un éditeur de liens est de rechercher et de localiser des modules / routines référencés dans un programme et de déterminer l'emplacement de mémoire où ces codes seront chargés, faisant que l'instruction du programme ait des références absolues.
Chargeur
Loader fait partie du système d'exploitation et est responsable du chargement des fichiers exécutables en mémoire et de leur exécution. Il calcule la taille d'un programme (instructions et données) et crée de l'espace mémoire pour celui-ci. Il initialise divers registres pour lancer l'exécution.
Compilateur croisé
Un compilateur qui s'exécute sur la plateforme (A) et est capable de générer du code exécutable pour la plateforme (B) est appelé un compilateur croisé.
Compilateur source à source
Un compilateur qui prend le code source d'un langage de programmation et le traduit dans le code source d'un autre langage de programmation est appelé un compilateur source-à-source.
Architecture du compilateur
Un compilateur peut être divisé en deux phases en fonction de la façon dont ils compilent.
Phase d'analyse
Connu comme le frontal du compilateur, le analysis phase du compilateur lit le programme source, le divise en parties principales, puis vérifie les erreurs lexicales, grammaticales et syntaxiques.La phase d'analyse génère une représentation intermédiaire du programme source et de la table des symboles, qui doit être transmise à la phase de synthèse en entrée .
Phase de synthèse
Connu comme le back-end du compilateur, le synthesis phase génère le programme cible à l'aide d'une représentation de code source intermédiaire et d'une table de symboles.
Un compilateur peut avoir plusieurs phases et passes.
Pass : Une passe fait référence au parcours d'un compilateur dans tout le programme.
Phase: Une phase d'un compilateur est une étape distincte, qui prend une entrée de l'étape précédente, traite et produit une sortie qui peut être utilisée comme entrée pour l'étape suivante. Une passe peut avoir plus d'une phase.
Phases du compilateur
Le processus de compilation est une séquence de différentes phases. Chaque phase prend l'entrée de son étape précédente, a sa propre représentation du programme source et transmet sa sortie à la phase suivante du compilateur. Laissez-nous comprendre les phases d'un compilateur.
Analyse lexicale
La première phase du scanner fonctionne comme un scanner de texte. Cette phase analyse le code source sous forme de flux de caractères et le convertit en lexèmes significatifs. L'analyseur lexical représente ces lexèmes sous forme de jetons comme:
<token-name, attribute-value>
Analyse de la syntaxe
La phase suivante est appelée l'analyse syntaxique ou parsing. Il prend le jeton produit par l'analyse lexicale comme entrée et génère un arbre d'analyse (ou arbre de syntaxe). Dans cette phase, les dispositions des jetons sont vérifiées par rapport à la grammaire du code source, c'est-à-dire que l'analyseur vérifie si l'expression faite par les jetons est syntaxiquement correcte.
Analyse sémantique
L'analyse sémantique vérifie si l'arbre d'analyse construit suit les règles du langage. Par exemple, l'attribution de valeurs se fait entre des types de données compatibles et l'ajout d'une chaîne à un entier. En outre, l'analyseur sémantique garde une trace des identifiants, de leurs types et expressions; si les identificateurs sont déclarés avant utilisation ou non, etc. L'analyseur sémantique produit un arbre de syntaxe annoté en sortie.
Génération de code intermédiaire
Après analyse sémantique, le compilateur génère un code intermédiaire du code source pour la machine cible. Il représente un programme pour une machine abstraite. Il se situe entre le langage de haut niveau et le langage machine. Ce code intermédiaire doit être généré de manière à faciliter sa traduction dans le code machine cible.
Optimisation du code
La phase suivante consiste à optimiser le code du code intermédiaire. L'optimisation peut être considérée comme quelque chose qui supprime les lignes de code inutiles et organise la séquence d'instructions afin d'accélérer l'exécution du programme sans gaspiller de ressources (CPU, mémoire).
Génération de code
Dans cette phase, le générateur de code prend la représentation optimisée du code intermédiaire et la mappe au langage machine cible. Le générateur de code traduit le code intermédiaire en une séquence de code machine (généralement) relocalisable. La séquence d'instructions du code machine exécute la tâche comme le ferait le code intermédiaire.
Table des symboles
C'est une structure de données maintenue pendant toutes les phases d'un compilateur. Tous les noms d'identifiant ainsi que leurs types sont stockés ici. La table des symboles permet au compilateur de rechercher rapidement l'enregistrement d'identificateur et de le récupérer. La table des symboles est également utilisée pour la gestion du périmètre.
L'analyse lexicale est la première phase d'un compilateur. Il prend le code source modifié des préprocesseurs de langage qui sont écrits sous forme de phrases. L'analyseur lexical décompose ces syntaxes en une série de jetons, en supprimant tout espace ou commentaire dans le code source.
Si l'analyseur lexical trouve un jeton invalide, il génère une erreur. L'analyseur lexical travaille en étroite collaboration avec l'analyseur de syntaxe. Il lit les flux de caractères du code source, vérifie les jetons légaux et transmet les données à l'analyseur de syntaxe lorsqu'il le demande.
Jetons
On dit que les lexèmes sont une séquence de caractères (alphanumériques) dans un jeton. Il existe des règles prédéfinies pour que chaque lexème soit identifié comme un jeton valide. Ces règles sont définies par des règles de grammaire, au moyen d'un modèle. Un modèle explique ce qui peut être un jeton, et ces modèles sont définis au moyen d'expressions régulières.
Dans le langage de programmation, les mots-clés, les constantes, les identificateurs, les chaînes, les nombres, les opérateurs et les symboles de ponctuation peuvent être considérés comme des jetons.
Par exemple, en langage C, la ligne de déclaration de variable
int value = 100;
contient les jetons:
int (keyword), value (identifier), = (operator), 100 (constant) and ; (symbol).
Spécifications des jetons
Comprenons comment la théorie du langage entreprend les termes suivants:
Alphabets
Tout ensemble fini de symboles {0,1} est un ensemble d'alphabets binaires, {0,1,2,3,4,5,6,7,8,9, A, B, C, D, E, F} est un ensemble d'alphabets hexadécimaux, {az, AZ} est un ensemble d'alphabets de langue anglaise.
Cordes
Toute séquence finie d'alphabets est appelée une chaîne. La longueur de la chaîne est le nombre total d'occurrences d'alphabets, par exemple, la longueur de la chaîne tutorialspoint est de 14 et est désignée par | tutorialspoint | = 14. Une chaîne sans alphabets, c'est-à-dire une chaîne de longueur nulle est appelée chaîne vide et est notée ε (epsilon).
Symboles spéciaux
Un langage de haut niveau typique contient les symboles suivants: -
Symboles arithmétiques | Addition (+), Soustraction (-), Modulo (%), Multiplication (*), Division (/) |
Ponctuation | Virgule (,), Point-virgule (;), Point (.), Flèche (->) |
Affectation | = |
Affectation spéciale | + =, / =, * =, - = |
Comparaison | ==,! =, <, <=,>,> = |
Préprocesseur | # |
Spécificateur d'emplacement | & |
Logique | &, &&, |, ||,! |
Opérateur de quart | >>, >>>, <<, <<< |
Langue
Une langue est considérée comme un ensemble fini de chaînes sur un ensemble fini d'alphabets. Les langages informatiques sont considérés comme des ensembles finis et des opérations définies mathématiquement peuvent être effectuées sur eux. Les langages finis peuvent être décrits au moyen d'expressions régulières.
Expressions régulières
L'analyseur lexical doit scanner et identifier uniquement un ensemble fini de chaîne / jeton / lexème valide appartenant à la langue en cours. Il recherche le modèle défini par les règles de langue.
Les expressions régulières ont la capacité d'exprimer des langages finis en définissant un modèle pour des chaînes finies de symboles. La grammaire définie par les expressions régulières est appeléeregular grammar. Le langage défini par la grammaire régulière est appeléregular language.
L'expression régulière est une notation importante pour spécifier des modèles. Chaque modèle correspond à un ensemble de chaînes, de sorte que les expressions régulières servent de noms pour un ensemble de chaînes. Les jetons de langage de programmation peuvent être décrits par des langages normaux. La spécification d'expressions régulières est un exemple de définition récursive. Les langues ordinaires sont faciles à comprendre et ont une mise en œuvre efficace.
Il existe un certain nombre de lois algébriques auxquelles obéissent les expressions régulières, qui peuvent être utilisées pour manipuler des expressions régulières sous des formes équivalentes.
Opérations
Les différentes opérations sur les langues sont:
L'union de deux langues L et M s'écrit
LUM = {s | s est dans L ou s est dans M}
La concaténation de deux langues L et M s'écrit
LM = {st | s est dans L et t est dans M}
La fermeture de Kleene d'une langue L s'écrit
L * = Zéro occurrence ou plus de la langue L.
Notations
Si r et s sont des expressions régulières désignant les langages L (r) et L (s), alors
Union : (r) | (s) est une expression régulière désignant L (r) UL (s)
Concatenation : (r) (s) est une expression régulière désignant L (r) L (s)
Kleene closure : (r) * est une expression régulière désignant (L (r)) *
(r) est une expression régulière désignant L (r)
Préséance et associativité
- *, concaténation (.) et | (signe de tuyau) sont laissés associatifs
- * a la priorité la plus élevée
- La concaténation (.) A la deuxième priorité la plus élevée.
- | (signe de tuyau) a la priorité la plus basse de tous.
Représenter des jetons valides d'un langage dans une expression régulière
Si x est une expression régulière, alors:
x * signifie zéro occurrence ou plus de x.
c'est-à-dire qu'il peut générer {e, x, xx, xxx, xxxx,…}
x + signifie une ou plusieurs occurrences de x.
c'est-à-dire qu'il peut générer {x, xx, xxx, xxxx…} ou xx *
X? signifie au plus une occurrence de x
c'est-à-dire qu'il peut générer {x} ou {e}.
[az] est tous les alphabets minuscules de la langue anglaise.
[AZ] est tous les alphabets majuscules de la langue anglaise.
[0-9] est tous les chiffres naturels utilisés en mathématiques.
Représenter l'occurrence de symboles à l'aide d'expressions régulières
lettre = [a - z] ou [A - Z]
chiffre = 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 ou [0-9]
signe = [+ | -]
Représenter des jetons de langage à l'aide d'expressions régulières
Décimal = (signe) ? (chiffre) +
Identifiant = (lettre) (lettre | chiffre) *
Le seul problème qui reste avec l'analyseur lexical est de savoir comment vérifier la validité d'une expression régulière utilisée pour spécifier les modèles de mots-clés d'une langue. Une solution bien acceptée consiste à utiliser des automates finis pour la vérification.
Automates finis
Les automates finis sont une machine à états qui prend une chaîne de symboles en entrée et change son état en conséquence. Les automates finis sont un outil de reconnaissance des expressions régulières. Lorsqu'une chaîne d'expression régulière est introduite dans des automates finis, elle change son état pour chaque littéral. Si la chaîne d'entrée est traitée avec succès et que l'automate atteint son état final, elle est acceptée, c'est-à-dire que la chaîne qui vient d'être alimentée est considérée comme un jeton valide de la langue en cours.
Le modèle mathématique des automates finis se compose de:
- Ensemble fini d'états (Q)
- Ensemble fini de symboles d'entrée (Σ)
- Un état de démarrage (q0)
- Ensemble d'états finaux (qf)
- Fonction de transition (δ)
La fonction de transition (δ) mappe l'ensemble fini d'états (Q) à un ensemble fini de symboles d'entrée (Σ), Q × Σ ➔ Q
Construction finie d'automates
Soit L (r) un langage régulier reconnu par certains automates finis (FA).
States: Les états de FA sont représentés par des cercles. Les noms d'état sont écrits à l'intérieur de cercles.
Start state: L'état à partir duquel les automates commencent, est appelé état de départ. L'état de démarrage a une flèche pointée vers lui.
Intermediate states: Tous les états intermédiaires ont au moins deux flèches; un pointant vers et un autre pointant d'eux.
Final state: Si la chaîne d'entrée est analysée avec succès, les automates devraient être dans cet état. L'état final est représenté par des doubles cercles. Il peut avoir n'importe quel nombre impair de flèches pointant vers lui et un nombre pair de flèches pointant vers lui. Le nombre de flèches impaires est supérieur à pair, c'est-à-direodd = even+1.
Transition: La transition d'un état à un autre état se produit lorsqu'un symbole souhaité dans l'entrée est trouvé. Lors de la transition, les automates peuvent passer à l'état suivant ou rester dans le même état. Le mouvement d'un état à un autre est représenté par une flèche dirigée, où les flèches indiquent l'état de destination. Si les automates restent dans le même état, une flèche pointant d'un état vers lui-même est dessinée.
Example: Nous supposons que FA accepte toute valeur binaire à trois chiffres se terminant par le chiffre 1. FA = {Q (q 0 , q f ), Σ (0,1), q 0 , q f , δ}
Règle de correspondance la plus longue
Lorsque l'analyseur lexical lit le code source, il scanne le code lettre par lettre; et lorsqu'il rencontre un espace, un symbole d'opérateur ou des symboles spéciaux, il décide qu'un mot est terminé.
For example:
int intvalue;
En balayant les deux lexèmes jusqu'à «int», l'analyseur lexical ne peut pas déterminer s'il s'agit d'un mot-clé int ou des initiales de l'identifiant int value.
La règle de correspondance la plus longue stipule que le lexème analysé doit être déterminé en fonction de la correspondance la plus longue parmi tous les jetons disponibles.
L'analyseur lexical suit également rule priorityoù un mot réservé, par exemple un mot-clé, d'une langue a la priorité sur l'entrée de l'utilisateur. Autrement dit, si l'analyseur lexical trouve un lexème qui correspond à n'importe quel mot réservé existant, il devrait générer une erreur.
L'analyse ou l'analyse syntaxique est la deuxième phase d'un compilateur. Dans ce chapitre, nous allons apprendre les concepts de base utilisés dans la construction d'un analyseur syntaxique.
Nous avons vu qu'un analyseur lexical peut identifier des jetons à l'aide d'expressions régulières et de règles de modèle. Mais un analyseur lexical ne peut pas vérifier la syntaxe d'une phrase donnée en raison des limitations des expressions régulières. Les expressions régulières ne peuvent pas vérifier les jetons d'équilibrage, tels que les parenthèses. Par conséquent, cette phase utilise la grammaire sans contexte (CFG), qui est reconnue par les automates push-down.
CFG, en revanche, est un sur-ensemble de la grammaire régulière, comme illustré ci-dessous:
Cela implique que chaque grammaire régulière est également sans contexte, mais il existe des problèmes qui dépassent le cadre de la grammaire régulière. CFG est un outil utile pour décrire la syntaxe des langages de programmation.
Grammaire sans contexte
Dans cette section, nous allons d'abord voir la définition de la grammaire sans contexte et introduire les terminologies utilisées dans la technologie d'analyse.
Une grammaire sans contexte comporte quatre composants:
Un ensemble de non-terminals(V). Les non-terminaux sont des variables syntaxiques qui désignent des ensembles de chaînes. Les non-terminaux définissent des ensembles de chaînes qui aident à définir le langage généré par la grammaire.
Un ensemble de jetons, appelé terminal symbols(Σ). Les terminaux sont les symboles de base à partir desquels les chaînes sont formées.
Un ensemble de productions(P). Les productions d'une grammaire précisent la manière dont les terminaux et les non-terminaux peuvent être combinés pour former des chaînes. Chaque production se compose d'unnon-terminal appelé le côté gauche de la production, une flèche et une séquence de jetons et / ou on- terminals, a appelé le côté droit de la production.
L'un des non-terminaux est désigné comme le symbole de départ (S); d'où commence la production.
Les chaînes sont dérivées du symbole de début en remplaçant à plusieurs reprises un non-terminal (initialement le symbole de départ) par le côté droit d'une production, pour ce non-terminal.
Exemple
Nous prenons le problème du langage palindrome, qui ne peut être décrit au moyen de l'expression régulière. Autrement dit, L = {w | w = w R } n'est pas un langage régulier. Mais il peut être décrit au moyen de CFG, comme illustré ci-dessous:
G = ( V, Σ, P, S )
Où:
V = { Q, Z, N }
Σ = { 0, 1 }
P = { Q → Z | Q → N | Q → ℇ | Z → 0Q0 | N → 1Q1 }
S = { Q }
Cette grammaire décrit le langage palindrome, tel que: 1001, 11100111, 00100, 1010101, 11111, etc.
Analyseurs de syntaxe
Un analyseur de syntaxe ou un parseur prend l'entrée d'un analyseur lexical sous la forme de flux de jetons. L'analyseur analyse le code source (jeton stream) par rapport aux règles de production pour détecter toute erreur dans le code. La sortie de cette phase est unparse tree.
De cette façon, l'analyseur accomplit deux tâches, à savoir, analyser le code, rechercher les erreurs et générer un arbre d'analyse comme sortie de la phase.
On s'attend à ce que les analyseurs analysent tout le code même si des erreurs existent dans le programme. Les analyseurs utilisent des stratégies de récupération d'erreur, que nous apprendrons plus loin dans ce chapitre.
Dérivation
Une dérivation est essentiellement une séquence de règles de production, afin d'obtenir la chaîne d'entrée. Au cours de l'analyse, nous prenons deux décisions pour une forme d'entrée sententielle:
- Décider du non-terminal qui doit être remplacé.
- Décider de la règle de production par laquelle le non-terminal sera remplacé.
Pour décider du non-terminal à remplacer par la règle de production, nous pouvons avoir deux options.
Dérivation la plus à gauche
Si la forme sententielle d'une entrée est scannée et remplacée de gauche à droite, elle est appelée dérivation la plus à gauche. La forme sententielle dérivée par la dérivation la plus à gauche est appelée la forme sententielle gauche.
Dérivation la plus à droite
Si nous analysons et remplaçons l'entrée par des règles de production, de droite à gauche, on parle de dérivation la plus à droite. La forme sententielle dérivée de la dérivation la plus à droite est appelée la forme sententielle droite.
Example
Règles de production:
E → E + E
E → E * E
E → id
Chaîne d'entrée: id + id * id
La dérivation la plus à gauche est:
E → E * E
E → E + E * E
E → id + E * E
E → id + id * E
E → id + id * id
Notez que le côté non terminal le plus à gauche est toujours traité en premier.
La dérivation la plus à droite est:
E → E + E
E → E + E * E
E → E + E * id
E → E + id * id
E → id + id * id
Analyser l'arbre
Un arbre d'analyse est une représentation graphique d'une dérivation. Il est pratique de voir comment les chaînes sont dérivées du symbole de début. Le symbole de début de la dérivation devient la racine de l'arborescence d'analyse. Voyons cela par un exemple du dernier sujet.
Nous prenons la dérivation la plus à gauche de a + b * c
La dérivation la plus à gauche est:
E → E * E
E → E + E * E
E → id + E * E
E → id + id * E
E → id + id * id
Étape 1:
E → E * E |
|
Étape 2:
E → E + E * E |
|
Étape 3:
E → id + E * E |
|
Étape 4:
E → id + id * E |
|
Étape 5:
E → id + id * id |
|
Dans un arbre d'analyse:
- Tous les nœuds feuilles sont des terminaux.
- Tous les nœuds intérieurs sont des non-terminaux.
- Le parcours dans l'ordre donne la chaîne d'entrée d'origine.
Un arbre d'analyse décrit l'associativité et la priorité des opérateurs. Le sous-arbre le plus profond est parcouru en premier, donc l'opérateur de ce sous-arbre a la priorité sur l'opérateur qui se trouve dans les nœuds parents.
Types d'analyse
Les analyseurs de syntaxe suivent des règles de production définies au moyen d'une grammaire sans contexte. La façon dont les règles de production sont implémentées (dérivation) divise l'analyse en deux types: l'analyse descendante et l'analyse ascendante.
Analyse descendante
Lorsque l'analyseur commence à construire l'arborescence d'analyse à partir du symbole de début, puis essaie de transformer le symbole de début en entrée, cela s'appelle une analyse descendante.
Recursive descent parsing: C'est une forme courante d'analyse descendante. Il est appelé récursif car il utilise des procédures récursives pour traiter l'entrée. L'analyse des descentes récursives souffre d'un retour en arrière.
Backtracking: Cela signifie que si une dérivation d'une production échoue, l'analyseur de syntaxe redémarre le processus en utilisant différentes règles de la même production. Cette technique peut traiter la chaîne d'entrée plusieurs fois pour déterminer la bonne production.
Analyse ascendante
Comme son nom l'indique, l'analyse ascendante commence par les symboles d'entrée et tente de construire l'arborescence d'analyse jusqu'au symbole de départ.
Example:
Chaîne d'entrée: a + b * c
Règles de production:
S → E
E → E + T
E → E * T
E → T
T → id
Commençons l'analyse ascendante
a + b * c
Lisez l'entrée et vérifiez si une production correspond à l'entrée:
a + b * c
T + b * c
E + b * c
E + T * c
E * c
E * T
E
S
Ambiguïté
Une grammaire G est dite ambiguë si elle a plus d'un arbre d'analyse (dérivation gauche ou droite) pour au moins une chaîne.
Example
E → E + E
E → E – E
E → id
Pour la chaîne id + id - id, la grammaire ci-dessus génère deux arbres d'analyse:
On dit que le langage généré par une grammaire ambiguë est inherently ambiguous. L'ambiguïté dans la grammaire n'est pas bonne pour la construction d'un compilateur. Aucune méthode ne peut détecter et supprimer l'ambiguïté automatiquement, mais elle peut être supprimée soit en réécrivant la grammaire entière sans ambiguïté, soit en définissant et en respectant les contraintes d'associativité et de précédence.
Associativité
Si un opérande a des opérateurs des deux côtés, le côté sur lequel l'opérateur prend cet opérande est décidé par l'associativité de ces opérateurs. Si l'opération est associative à gauche, l'opérande sera pris par l'opérateur de gauche ou si l'opération est associative à droite, l'opérateur de droite prendra l'opérande.
Example
Les opérations telles que l'addition, la multiplication, la soustraction et la division restent associatives. Si l'expression contient:
id op id op id
il sera évalué comme:
(id op id) op id
Par exemple, (id + id) + id
Les opérations comme l'exponentiation sont associatives correctes, c'est-à-dire que l'ordre d'évaluation dans la même expression sera:
id op (id op id)
Par exemple, id ^ (id ^ id)
Priorité
Si deux opérateurs différents partagent un opérande commun, la priorité des opérateurs décide de celui qui prendra l'opérande. Autrement dit, 2 + 3 * 4 peut avoir deux arbres d'analyse différents, l'un correspondant à (2 + 3) * 4 et l'autre correspondant à 2+ (3 * 4). En définissant la priorité parmi les opérateurs, ce problème peut être facilement supprimé. Comme dans l'exemple précédent, mathématiquement * (multiplication) a priorité sur + (addition), donc l'expression 2 + 3 * 4 sera toujours interprétée comme:
2 + (3 * 4)
Ces méthodes réduisent les risques d'ambiguïté dans une langue ou sa grammaire.
Récursivité gauche
Une grammaire devient récursive à gauche si elle a un 'A' non terminal dont la dérivation contient 'A' lui-même comme symbole le plus à gauche. La grammaire récursive gauche est considérée comme une situation problématique pour les analyseurs de haut en bas. Les analyseurs de haut en bas commencent l'analyse à partir du symbole de début, qui en soi n'est pas terminal. Ainsi, lorsque l'analyseur rencontre le même non-terminal dans sa dérivation, il devient difficile pour lui de juger quand arrêter l'analyse du non-terminal gauche et il entre dans une boucle infinie.
Example:
(1) A => Aα | β
(2) S => Aα | β
A => Sd
(1) est un exemple de récursion immédiate à gauche, où A est un symbole non terminal et α représente une chaîne de non terminaux.
(2) est un exemple de récursion indirecte à gauche.
Un analyseur de haut en bas analysera d'abord le A, qui à son tour produira une chaîne composée de A lui-même et l'analyseur peut entrer dans une boucle pour toujours.
Suppression de la récursivité gauche
Une façon de supprimer la récursivité à gauche consiste à utiliser la technique suivante:
La production
A => Aα | β
est converti en productions suivantes
A => βA’
A => αA’ | ε
Cela n'affecte pas les chaînes dérivées de la grammaire, mais supprime la récursivité gauche immédiate.
La deuxième méthode consiste à utiliser l'algorithme suivant, qui devrait éliminer toutes les récursions directes et indirectes à gauche.
Algorithm
START
Arrange non-terminals in some order like A1, A2, A3,…, An
for each i from 1 to n
{
for each j from 1 to i-1
{
replace each production of form Ai⟹Aj
with Ai ⟹ δ1 | δ2 | δ3 |…|
where Aj ⟹ δ1 | δ2|…| δn are current Aj productions
}
}
eliminate immediate left-recursion
END
Example
L'ensemble de production
S => Aα | β
A => Sd
après avoir appliqué l'algorithme ci-dessus, devrait devenir
S => Aα | β
A => Aαd | βd
puis supprimez la récursivité gauche immédiate en utilisant la première technique.
A => βdA’
A => αdA’ | ε
Désormais, aucune partie de la production n'a de récursion gauche directe ou indirecte.
Affacturage gauche
Si plusieurs règles de production de grammaire ont une chaîne de préfixe commune, alors l'analyseur de haut en bas ne peut pas faire un choix quant à la production à prendre pour analyser la chaîne en cours.
Example
Si un analyseur de haut en bas rencontre une production comme
A ⟹ αβ | α | …
Ensuite, il ne peut pas déterminer quelle production suivre pour analyser la chaîne car les deux productions partent du même terminal (ou non terminal). Pour supprimer cette confusion, nous utilisons une technique appelée affacturage gauche.
La factorisation de gauche transforme la grammaire pour la rendre utile pour les analyseurs de haut en bas. Dans cette technique, nous faisons une production pour chaque préfixe commun et le reste de la dérivation est ajouté par de nouvelles productions.
Example
Les productions ci-dessus peuvent être écrites comme
A => αA’
A’=> β | | …
Désormais, l'analyseur n'a qu'une seule production par préfixe, ce qui facilite la prise de décision.
Ensembles premier et suivi
Une partie importante de la construction de la table d'analyseur consiste à créer les premiers ensembles et les suivants. Ces ensembles peuvent fournir la position réelle de n'importe quel terminal dans la dérivation. Ceci est fait pour créer la table d'analyse où la décision de remplacer T [A, t] = α par une règle de production.
Premier set
Cet ensemble est créé pour savoir quel symbole de terminal est dérivé en première position par un non-terminal. Par exemple,
α → t β
C'est-à-dire que α dérive t (terminal) dans la toute première position. Donc, t ∈ PREMIER (α).
Algorithme de calcul du premier jeu
Regardez la définition de FIRST (α) set:
- si α est un terminal, alors FIRST (α) = {α}.
- si α est un non-terminal et α → ℇ est une production, alors FIRST (α) = {ℇ}.
- si α est un non-terminal et α → 1 2 3… n et tout FIRST () contient t alors t est dans FIRST (α).
Le premier ensemble peut être vu comme: PREMIER (α) = {t | α → * t β} ∪ {ℇ | α → * ε}
Suivez l'ensemble
De même, nous calculons quel symbole terminal suit immédiatement un α non terminal dans les règles de production. Nous ne considérons pas ce que le non-terminal peut générer mais au contraire, nous voyons quel serait le prochain symbole terminal qui suivrait les productions d'un non-terminal.
Algorithme de calcul de l'ensemble de suivi:
si α est un symbole de départ, alors FOLLOW () = $
si α est un non-terminal et a une production α → AB, alors FIRST (B) est dans FOLLOW (A) sauf ℇ.
si α est un non-terminal et a une production α → AB, où B ℇ, alors FOLLOW (A) est dans FOLLOW (α).
L'ensemble de suivi peut être vu comme suit: FOLLOW (α) = {t | S * αt *}
Stratégies de récupération d'erreur
Un analyseur devrait être capable de détecter et de signaler toute erreur dans le programme. Il est prévu que lorsqu'une erreur est rencontrée, l'analyseur devrait être en mesure de la gérer et de continuer à analyser le reste de l'entrée. La plupart du temps, il est attendu de l'analyseur pour vérifier les erreurs, mais des erreurs peuvent survenir à différentes étapes du processus de compilation. Un programme peut présenter les types d'erreurs suivants à différentes étapes:
Lexical : nom d'un identifiant mal saisi
Syntactical : point-virgule manquant ou parenthèse non équilibrée
Semantical : attribution de valeur incompatible
Logical : code non accessible, boucle infinie
Il existe quatre stratégies de récupération d'erreur courantes qui peuvent être implémentées dans l'analyseur pour traiter les erreurs dans le code.
Mode panique
Lorsqu'un analyseur rencontre une erreur n'importe où dans l'instruction, il ignore le reste de l'instruction en ne traitant pas l'entrée d'une entrée erronée vers un délimiteur, tel qu'un point-virgule. C'est le moyen le plus simple de récupérer les erreurs et empêche également l'analyseur de développer des boucles infinies.
Mode déclaration
Lorsqu'un analyseur rencontre une erreur, il essaie de prendre des mesures correctives afin que le reste des entrées de l'instruction lui permette d'analyser en avant. Par exemple, insérer un point-virgule manquant, remplacer la virgule par un point-virgule, etc. Les concepteurs d'analyseurs doivent être prudents ici car une mauvaise correction peut conduire à une boucle infinie.
Productions d'erreur
Certaines erreurs courantes sont connues des concepteurs du compilateur qui peuvent se produire dans le code. De plus, les concepteurs peuvent créer une grammaire augmentée à utiliser, comme des productions qui génèrent des constructions erronées lorsque ces erreurs sont rencontrées.
Correction globale
L'analyseur considère le programme en cours dans son ensemble et essaie de comprendre ce que le programme est censé faire et essaie de trouver une correspondance la plus proche pour lui, ce qui est sans erreur. Lorsqu'une entrée erronée (instruction) X est alimentée, elle crée un arbre d'analyse pour une instruction Y sans erreur la plus proche. Cela peut permettre à l'analyseur de faire des changements minimes dans le code source, mais en raison de la complexité (temps et espace) de cette stratégie, elle n’a pas encore été mise en œuvre dans la pratique.
Arbres de syntaxe abstraite
Les représentations d'arbre d'analyse ne sont pas faciles à analyser par le compilateur, car elles contiennent plus de détails que nécessaire. Prenons comme exemple l'arborescence d'analyse suivante:
Si on regarde de près, nous trouvons que la plupart des nœuds feuilles sont des enfants uniques à leurs nœuds parents. Ces informations peuvent être éliminées avant de les transmettre à la phase suivante. En masquant des informations supplémentaires, nous pouvons obtenir un arbre comme indiqué ci-dessous:
L'arbre abstrait peut être représenté comme:
Les AST sont des structures de données importantes dans un compilateur avec le moins d'informations inutiles. Les AST sont plus compacts qu'un arbre d'analyse et peuvent être facilement utilisés par un compilateur.
Limitations des analyseurs de syntaxe
Les analyseurs de syntaxe reçoivent leurs entrées, sous forme de jetons, des analyseurs lexicaux. Les analyseurs lexicaux sont responsables de la validité d'un token fourni par l'analyseur de syntaxe. Les analyseurs de syntaxe présentent les inconvénients suivants:
- il ne peut pas déterminer si un jeton est valide,
- il ne peut pas déterminer si un jeton est déclaré avant d'être utilisé,
- il ne peut pas déterminer si un jeton est initialisé avant d'être utilisé,
- il ne peut pas déterminer si une opération effectuée sur un type de jeton est valide ou non.
Ces tâches sont accomplies par l'analyseur sémantique, que nous étudierons dans l'analyse sémantique.
Nous avons appris comment un analyseur construit des arbres d'analyse dans la phase d'analyse syntaxique. L'arbre d'analyse simple construit dans cette phase n'est généralement d'aucune utilité pour un compilateur, car il ne contient aucune information sur la façon d'évaluer l'arbre. Les productions de grammaire sans contexte, qui fait les règles du langage, ne permettent pas de les interpréter.
Par exemple
E → E + T
La production de CFG ci-dessus n'a pas de règle sémantique associée, et elle ne peut pas aider à donner un sens à la production.
Sémantique
La sémantique d'un langage donne un sens à ses constructions, comme les jetons et la structure syntaxique. La sémantique aide à interpréter les symboles, leurs types et leurs relations les uns avec les autres. L'analyse sémantique juge si la structure syntaxique construite dans le programme source en tire une signification ou non.
CFG + semantic rules = Syntax Directed Definitions
Par exemple:
int a = “value”;
ne doit pas émettre d'erreur dans la phase d'analyse lexicale et syntaxique, car il est lexicalement et structurellement correct, mais il doit générer une erreur sémantique car le type d'affectation diffère. Ces règles sont fixées par la grammaire de la langue et évaluées en analyse sémantique. Les tâches suivantes doivent être effectuées en analyse sémantique:
- Résolution de la portée
- Vérification de type
- Vérification liée à la baie
Erreurs sémantiques
Nous avons mentionné certaines des erreurs sémantiques que l'analyseur sémantique devrait reconnaître:
- Incompatibilité de type
- Variable non déclarée
- Utilisation abusive de l'identifiant réservé.
- Déclaration multiple de variable dans une portée.
- Accéder à une variable hors de portée.
- Inadéquation des paramètres réels et formels.
Grammaire des attributs
La grammaire des attributs est une forme spéciale de grammaire sans contexte où des informations supplémentaires (attributs) sont ajoutées à un ou plusieurs de ses non-terminaux afin de fournir des informations contextuelles. Chaque attribut a un domaine de valeurs bien défini, comme un entier, un flottant, un caractère, une chaîne et des expressions.
La grammaire des attributs est un moyen de fournir une sémantique à la grammaire sans contexte et elle peut aider à spécifier la syntaxe et la sémantique d'un langage de programmation. La grammaire des attributs (lorsqu'elle est vue comme un arbre d'analyse) peut transmettre des valeurs ou des informations entre les nœuds d'un arbre.
Example:
E → E + T { E.value = E.value + T.value }
La partie droite du CFG contient les règles sémantiques qui spécifient comment la grammaire doit être interprétée. Ici, les valeurs des non-terminaux E et T sont additionnées et le résultat est copié dans le non-terminal E.
Les attributs sémantiques peuvent être attribués à leurs valeurs à partir de leur domaine au moment de l'analyse et évalués au moment de l'attribution ou des conditions. En fonction de la manière dont les attributs obtiennent leurs valeurs, ils peuvent être globalement divisés en deux catégories: les attributs synthétisés et les attributs hérités.
Attributs synthétisés
Ces attributs obtiennent des valeurs à partir des valeurs d'attribut de leurs nœuds enfants. Pour illustrer, supposons la production suivante:
S → ABC
Si S prend des valeurs de ses nœuds enfants (A, B, C), alors on dit qu'il s'agit d'un attribut synthétisé, car les valeurs de ABC sont synthétisées en S.
Comme dans notre exemple précédent (E → E + T), le nœud parent E tire sa valeur de son nœud enfant. Les attributs synthétisés ne prennent jamais les valeurs de leurs nœuds parents ou des nœuds frères.
Attributs hérités
Contrairement aux attributs synthétisés, les attributs hérités peuvent prendre des valeurs de parents et / ou de frères et sœurs. Comme dans la production suivante,
S → ABC
A peut obtenir des valeurs de S, B et C.B peut prendre des valeurs de S, A et C. De même, C peut prendre des valeurs de S, A et B.
Expansion : Lorsqu'un non-terminal est étendu aux terminaux selon une règle grammaticale
Reduction: Lorsqu'un terminal est réduit à son non-terminal correspondant selon les règles de grammaire. Les arbres de syntaxe sont analysés de haut en bas et de gauche à droite. Chaque fois qu'une réduction se produit, nous appliquons ses règles sémantiques correspondantes (actions).
L'analyse sémantique utilise les traductions dirigées par la syntaxe pour effectuer les tâches ci-dessus.
L'analyseur sémantique reçoit AST (Abstract Syntax Tree) de son stade précédent (analyse syntaxique).
L'analyseur sémantique attache les informations d'attribut avec AST, qui sont appelées AST attribuées.
Les attributs sont deux valeurs de tuple, <nom d'attribut, valeur d'attribut>
Par exemple:
int value = 5;
<type, “integer”>
<presentvalue, “5”>
Pour chaque production, nous attachons une règle sémantique.
SDT attribué S
Si un SDT utilise uniquement des attributs synthétisés, il est appelé SDT attribué par S. Ces attributs sont évalués à l'aide de SDT attribués par S dont les actions sémantiques sont écrites après la production (côté droit).
Comme illustré ci-dessus, les attributs des SDT attribués par S sont évalués dans une analyse ascendante, car les valeurs des nœuds parents dépendent des valeurs des nœuds enfants.
SDT attribué par L
Cette forme de SDT utilise à la fois des attributs synthétisés et hérités avec la restriction de ne pas prendre les valeurs des bons frères.
Dans les SDT attribués en L, un non-terminal peut obtenir des valeurs de ses nœuds parent, enfant et frère. Comme dans la production suivante
S → ABC
S peut prendre des valeurs de A, B et C (synthétisées). A peut prendre des valeurs de S uniquement. B peut prendre des valeurs de S et A. C peut obtenir des valeurs de S, A et B. Aucun non-terminal ne peut obtenir des valeurs du frère à sa droite.
Les attributs des SDT attribués par L sont évalués par analyse en profondeur d'abord et de gauche à droite.
Nous pouvons conclure que si une définition est attribuée en S, alors elle est également attribuée en L car la définition attribuée en L englobe les définitions attribuées en S.
Dans le chapitre précédent, nous avons compris les concepts de base impliqués dans l'analyse. Dans ce chapitre, nous allons apprendre les différents types de méthodes de construction d'analyseurs disponibles.
L'analyse peut être définie de haut en bas ou de bas en haut en fonction de la façon dont l'arborescence d'analyse est construite.
Analyse descendante
Nous avons appris dans le dernier chapitre que la technique d'analyse descendante analyse l'entrée et commence à construire un arbre d'analyse à partir du nœud racine en descendant progressivement vers les nœuds feuilles. Les types d'analyse descendante sont décrits ci-dessous:
Analyse de descente récursive
La descente récursive est une technique d'analyse descendante qui construit l'arbre d'analyse à partir du haut et l'entrée est lue de gauche à droite. Il utilise des procédures pour chaque entité terminale et non terminale. Cette technique d'analyse analyse récursivement l'entrée pour créer un arbre d'analyse, qui peut ou non nécessiter un retour arrière. Mais la grammaire qui lui est associée (si elle n'est pas prise en compte) ne peut éviter le retour en arrière. Une forme d'analyse de descente récursive qui ne nécessite aucun suivi arrière est appeléepredictive parsing.
Cette technique d'analyse est considérée comme récursive car elle utilise une grammaire sans contexte qui est de nature récursive.
Suivi de retour
Les analyseurs de haut en bas partent du nœud racine (symbole de début) et comparent la chaîne d'entrée aux règles de production pour les remplacer (le cas échéant). Pour comprendre cela, prenez l'exemple suivant de CFG:
S → rXd | rZd
X → oa | ea
Z → ai
Pour une chaîne d'entrée: read, un analyseur de haut en bas, se comportera comme ceci:
Il commencera par S des règles de production et fera correspondre son rendement à la lettre la plus à gauche de l'entrée, c'est-à-dire «r». La production même de S (S → rXd) y correspond. Ainsi, l'analyseur de haut en bas passe à la lettre d'entrée suivante (c'est-à-dire «e»). L'analyseur essaie d'étendre 'X' non-terminal et vérifie sa production à partir de la gauche (X → oa). Il ne correspond pas au symbole d'entrée suivant. Ainsi, l'analyseur descendant fait marche arrière pour obtenir la prochaine règle de production de X, (X → ea).
Maintenant, l'analyseur fait correspondre toutes les lettres d'entrée de manière ordonnée. La chaîne est acceptée.
|
|
|
|
Analyseur prédictif
L'analyseur prédictif est un analyseur de descente récursive, qui a la capacité de prédire quelle production doit être utilisée pour remplacer la chaîne d'entrée. L'analyseur prédictif ne souffre pas de retour en arrière.
Pour accomplir ses tâches, l'analyseur prédictif utilise un pointeur d'anticipation, qui pointe vers les symboles d'entrée suivants. Pour rendre l'analyseur de retour libre, l'analyseur prédictif impose des contraintes sur la grammaire et n'accepte qu'une classe de grammaire connue sous le nom de grammaire LL (k).
L'analyse prédictive utilise une pile et une table d'analyse pour analyser l'entrée et générer une arborescence d'analyse. La pile et l'entrée contiennent un symbole de fin$pour indiquer que la pile est vide et que l'entrée est consommée. L'analyseur se réfère à la table d'analyse pour prendre toute décision sur la combinaison des éléments d'entrée et de pile.
Dans l'analyse par descente récursive, l'analyseur peut avoir plus d'une production à choisir pour une seule instance d'entrée, tandis que dans l'analyseur prédictif, chaque étape a au plus une production à choisir. Il peut y avoir des cas où aucune production ne correspond à la chaîne d'entrée, ce qui entraîne l'échec de la procédure d'analyse.
Analyseur LL
Un analyseur LL accepte la grammaire LL. La grammaire LL est un sous-ensemble de la grammaire sans contexte mais avec quelques restrictions pour obtenir la version simplifiée, afin de réaliser une implémentation facile. La grammaire LL peut être implémentée au moyen des deux algorithmes à savoir, descente récursive ou pilotée par table.
L'analyseur LL est noté LL (k). Le premier L dans LL (k) analyse l'entrée de gauche à droite, le second L dans LL (k) représente la dérivation la plus à gauche et k lui-même représente le nombre d'anticipations. Généralement k = 1, donc LL (k) peut aussi s'écrire LL (1).
Algorithme d'analyse LL
Nous pouvons nous en tenir à LL (1) déterministe pour l'explication de l'analyseur, car la taille de la table croît exponentiellement avec la valeur de k. Deuxièmement, si une grammaire donnée n'est pas LL (1), alors généralement, ce n'est pas LL (k), pour tout k donné.
Ci-dessous est un algorithme pour l'analyse LL (1):
Input:
string ω
parsing table M for grammar G
Output:
If ω is in L(G) then left-most derivation of ω,
error otherwise.
Initial State : $S on stack (with S being start symbol) ω$ in the input buffer
SET ip to point the first symbol of ω$.
repeat
let X be the top stack symbol and a the symbol pointed by ip.
if X∈ Vt or $
if X = a
POP X and advance ip.
else
error()
endif
else /* X is non-terminal */
if M[X,a] = X → Y1, Y2,... Yk
POP X
PUSH Yk, Yk-1,... Y1 /* Y1 on top */
Output the production X → Y1, Y2,... Yk
else
error()
endif
endif
until X = $ /* empty stack */
Une grammaire G est LL (1) si A-> alpha | b sont deux productions distinctes de G:
pour aucun terminal, alpha et bêta dérivent des chaînes commençant par a.
au plus un des alpha et bêta peut dériver une chaîne vide.
si beta => t, alors alpha ne dérive aucune chaîne commençant par un terminal dans FOLLOW (A).
Analyse ascendante
L'analyse ascendante commence à partir des nœuds feuilles d'un arbre et fonctionne vers le haut jusqu'à ce qu'elle atteigne le nœud racine. Ici, on part d'une phrase puis on applique les règles de production de manière inverse pour atteindre le symbole de départ. L'image ci-dessous illustre les analyseurs ascendants disponibles.
Analyse Shift-Réduire
L'analyse de réduction de décalage utilise deux étapes uniques pour l'analyse ascendante. Ces étapes sont appelées étapes de décalage et réduction de pas.
Shift step: L'étape de décalage fait référence à l'avancement du pointeur d'entrée vers le symbole d'entrée suivant, appelé symbole décalé. Ce symbole est poussé sur la pile. Le symbole décalé est traité comme un nœud unique de l'arborescence d'analyse.
Reduce step: Lorsque l'analyseur trouve une règle de grammaire complète (RHS) et la remplace par (LHS), on parle de réduction de pas. Cela se produit lorsque le haut de la pile contient une poignée. Pour réduire, une fonction POP est exécutée sur la pile qui sort de la poignée et la remplace par le symbole non terminal LHS.
Analyseur LR
L'analyseur LR est un analyseur ascendant non récursif, à réduction de décalage. Il utilise une large classe de grammaire sans contexte, ce qui en fait la technique d'analyse syntaxique la plus efficace. Les analyseurs LR sont également connus sous le nom d'analyseurs LR (k), où L représente le balayage de gauche à droite du flux d'entrée; R représente la construction de la dérivation la plus à droite en sens inverse, et k désigne le nombre de symboles d'anticipation pour prendre des décisions.
Il existe trois algorithmes largement utilisés pour construire un analyseur LR:
- SLR (1) - Analyseur LR simple:
- Fonctionne sur la plus petite classe de grammaire
- Peu d'états, donc très petit tableau
- Construction simple et rapide
- LR (1) - Analyseur LR:
- Fonctionne sur un ensemble complet de grammaire LR (1)
- Génère une grande table et un grand nombre d'états
- Construction lente
- LALR (1) - Analyseur LR Look-Ahead:
- Fonctionne sur la taille intermédiaire de la grammaire
- Le nombre d'états est le même que dans SLR (1)
Algorithme d'analyse LR
Nous décrivons ici un algorithme squelette d'un analyseur LR:
token = next_token()
repeat forever
s = top of stack
if action[s, token] = “shift si” then
PUSH token
PUSH si
token = next_token()
else if action[s, tpken] = “reduce A::= β“ then
POP 2 * |β| symbols
s = top of stack
PUSH A
PUSH goto[s,A]
else if action[s, token] = “accept” then
return
else
error()
LL contre LR
LL | G / D |
---|---|
Fait une dérivation la plus à gauche. | Fait une dérivation la plus à droite en sens inverse. |
Commence par le non-terminal racine de la pile. | Se termine par le non-terminal racine de la pile. |
Se termine lorsque la pile est vide. | Commence avec une pile vide. |
Utilise la pile pour désigner ce qui reste à attendre. | Utilise la pile pour désigner ce qui est déjà vu. |
Construit l'arborescence d'analyse de haut en bas. | Construit l'arborescence d'analyse de bas en haut. |
Saute en continu un non-terminal hors de la pile et pousse le côté droit correspondant. | Tente de reconnaître un côté droit de la pile, le fait apparaître et pousse le non-terminal correspondant. |
Développe les non-terminaux. | Réduit les non-terminaux. |
Lit les terminaux lorsqu'il en sort un de la pile. | Lit les terminaux pendant qu'il les pousse sur la pile. |
Parcours de pré-commande de l'arbre d'analyse. | Parcours post-ordre de l'arbre d'analyse. |
Un programme en tant que code source est simplement une collection de texte (code, instructions, etc.) et pour le rendre vivant, il nécessite des actions à effectuer sur la machine cible. Un programme a besoin de ressources mémoire pour exécuter des instructions. Un programme contient des noms de procédures, d'identificateurs, etc., qui nécessitent un mappage avec l'emplacement de mémoire réel au moment de l'exécution.
Par runtime, nous entendons un programme en cours d'exécution. L'environnement d'exécution est un état de la machine cible, qui peut inclure des bibliothèques de logiciels, des variables d'environnement, etc., pour fournir des services aux processus exécutés dans le système.
Le système de support d'exécution est un package, principalement généré avec le programme exécutable lui-même et facilite la communication de processus entre le processus et l'environnement d'exécution. Il s'occupe de l'allocation et de la désallocation de la mémoire pendant l'exécution du programme.
Arbres d'activation
Un programme est une séquence d'instructions combinées en un certain nombre de procédures. Les instructions d'une procédure sont exécutées séquentiellement. Une procédure a un séparateur de début et de fin et tout ce qu'il contient est appelé le corps de la procédure. L'identifiant de procédure et la séquence d'instructions finies qu'il contient constituent le corps de la procédure.
L'exécution d'une procédure s'appelle son activation. Un enregistrement d'activation contient toutes les informations nécessaires pour appeler une procédure. Un enregistrement d'activation peut contenir les unités suivantes (selon la langue source utilisée).
Temporaires | Stocke les valeurs temporaires et intermédiaires d'une expression. |
Données locales | Stocke les données locales de la procédure appelée. |
État de la machine | Stocke l'état de la machine comme les registres, le compteur de programmes, etc., avant que la procédure ne soit appelée. |
Lien de contrôle | Stocke l'adresse d'enregistrement d'activation de la procédure de l'appelant. |
Lien d'accès | Stocke les informations des données qui sont en dehors de la portée locale. |
Paramètres réels | Stocke les paramètres réels, c'est-à-dire les paramètres utilisés pour envoyer des entrées à la procédure appelée. |
Valeur de retour | Stocke les valeurs de retour. |
Chaque fois qu'une procédure est exécutée, son enregistrement d'activation est stocké sur la pile, également appelée pile de contrôle. Lorsqu'une procédure appelle une autre procédure, l'exécution de l'appelant est suspendue jusqu'à la fin de l'exécution de la procédure appelée. A ce moment, l'enregistrement d'activation de la procédure appelée est stocké sur la pile.
On suppose que le contrôle du programme se déroule de manière séquentielle et lorsqu'une procédure est appelée, son contrôle est transféré à la procédure appelée. Lorsqu'une procédure appelée est exécutée, elle renvoie le contrôle à l'appelant. Ce type de flux de contrôle permet de représenter plus facilement une série d'activations sous la forme d'un arbre, appelé leactivation tree.
Pour comprendre ce concept, nous prenons un morceau de code comme exemple:
. . .
printf(“Enter Your Name: “);
scanf(“%s”, username);
show_data(username);
printf(“Press any key to continue…”);
. . .
int show_data(char *user)
{
printf(“Your name is %s”, username);
return 0;
}
. . .
Ci-dessous se trouve l'arbre d'activation du code donné.
Nous comprenons maintenant que les procédures sont exécutées en profondeur d'abord, donc l'allocation de pile est la meilleure forme de stockage appropriée pour les activations de procédure.
Allocation de stockage
L'environnement d'exécution gère les exigences de mémoire d'exécution pour les entités suivantes:
Code: Il s'agit de la partie texte d'un programme qui ne change pas à l'exécution. Ses besoins en mémoire sont connus au moment de la compilation.
Procedures: Leur partie texte est statique mais ils sont appelés de manière aléatoire. C'est pourquoi, le stockage en pile est utilisé pour gérer les appels de procédure et les activations.
Variables: Les variables ne sont connues qu'au moment de l'exécution, sauf si elles sont globales ou constantes. Le schéma d'allocation de mémoire de tas est utilisé pour gérer l'allocation et la désallocation de mémoire pour les variables au moment de l'exécution.
Allocation statique
Dans ce schéma d'allocation, les données de compilation sont liées à un emplacement fixe dans la mémoire et ne changent pas lorsque le programme s'exécute. Comme les besoins en mémoire et les emplacements de stockage sont connus à l'avance, le package de support d'exécution pour l'allocation et la désallocation de mémoire n'est pas nécessaire.
Allocation de pile
Les appels de procédure et leurs activations sont gérés au moyen de l'allocation de mémoire de pile. Il fonctionne dans la méthode du dernier entré, premier sorti (LIFO) et cette stratégie d'allocation est très utile pour les appels de procédure récursifs.
Allocation de tas
Les variables locales à une procédure sont allouées et désallouées uniquement lors de l'exécution. L'allocation de tas est utilisée pour allouer dynamiquement de la mémoire aux variables et la réclamer lorsque les variables ne sont plus nécessaires.
À l'exception de la zone de mémoire allouée statiquement, la mémoire de la pile et du tas peut augmenter et diminuer de manière dynamique et inattendue. Par conséquent, ils ne peuvent pas être fournis avec une quantité fixe de mémoire dans le système.
Comme le montre l'image ci-dessus, la partie texte du code se voit allouer une quantité fixe de mémoire. La pile et la mémoire de tas sont disposées aux extrêmes de la mémoire totale allouée au programme. Les deux rétrécissent et grandissent l'un contre l'autre.
Passage de paramètres
Le moyen de communication entre les procédures est appelé passage de paramètres. Les valeurs des variables d'une procédure appelante sont transférées à la procédure appelée par un mécanisme. Avant d'aller de l'avant, parcourez d'abord quelques terminologies de base relatives aux valeurs d'un programme.
valeur r
La valeur d'une expression est appelée sa valeur r. La valeur contenue dans une seule variable devient également une valeur r si elle apparaît sur le côté droit de l'opérateur d'affectation. Les valeurs r peuvent toujours être affectées à une autre variable.
valeur l
L'emplacement de la mémoire (adresse) où une expression est stockée est appelé valeur l de cette expression. Il apparaît toujours sur le côté gauche d'un opérateur d'affectation.
Par exemple:
day = 1;
week = day * 7;
month = 1;
year = month * 12;
À partir de cet exemple, nous comprenons que les valeurs constantes telles que 1, 7, 12 et des variables telles que jour, semaine, mois et année ont toutes des valeurs r. Seules les variables ont des valeurs l car elles représentent également l'emplacement mémoire qui leur est attribué.
Par exemple:
7 = x + y;
est une erreur de valeur l, car la constante 7 ne représente aucun emplacement mémoire.
Paramètres formels
Les variables qui prennent les informations transmises par la procédure de l'appelant sont appelées paramètres formels. Ces variables sont déclarées dans la définition de la fonction appelée.
Paramètres réels
Les variables dont les valeurs ou adresses sont transmises à la procédure appelée sont appelées paramètres réels. Ces variables sont spécifiées dans l'appel de fonction en tant qu'arguments.
Example:
fun_one()
{
int actual_parameter = 10;
call fun_two(int actual_parameter);
}
fun_two(int formal_parameter)
{
print formal_parameter;
}
Les paramètres formels contiennent les informations du paramètre réel, en fonction de la technique de passage de paramètre utilisée. Cela peut être une valeur ou une adresse.
Passer par valeur
Dans le mécanisme de passage par valeur, la procédure appelante transmet la valeur r des paramètres réels et le compilateur la place dans l'enregistrement d'activation de la procédure appelée. Les paramètres formels contiennent alors les valeurs transmises par la procédure appelante. Si les valeurs détenues par les paramètres formels sont modifiées, cela ne devrait avoir aucun impact sur les paramètres réels.
Passer par référence
Dans le mécanisme de passage par référence, la valeur l du paramètre réel est copiée dans l'enregistrement d'activation de la procédure appelée. De cette façon, la procédure appelée a maintenant l'adresse (emplacement mémoire) du paramètre réel et le paramètre formel fait référence au même emplacement mémoire. Par conséquent, si la valeur pointée par le paramètre formel est modifiée, l'impact doit être vu sur le paramètre réel car ils doivent également pointer vers la même valeur.
Passer par copie-restauration
Ce mécanisme de passage de paramètres fonctionne de manière similaire au «passage par référence», sauf que les modifications apportées aux paramètres réels sont effectuées à la fin de la procédure appelée. Lors de l'appel de fonction, les valeurs des paramètres réels sont copiées dans l'enregistrement d'activation de la procédure appelée. Les paramètres formels s'ils sont manipulés n'ont aucun effet en temps réel sur les paramètres réels (lorsque les valeurs l sont passées), mais lorsque la procédure appelée se termine, les valeurs l des paramètres formels sont copiées dans les valeurs l des paramètres réels.
Example:
int y;
calling_procedure()
{
y = 10;
copy_restore(y); //l-value of y is passed
printf y; //prints 99
}
copy_restore(int x)
{
x = 99; // y still has value 10 (unaffected)
y = 0; // y is now 0
}
Lorsque cette fonction se termine, la valeur l du paramètre formel x est copiée dans le paramètre réel y. Même si la valeur de y est modifiée avant la fin de la procédure, la valeur l de x est copiée dans la valeur l de y, ce qui lui permet de se comporter comme un appel par référence.
Passer par les noms
Des langages comme Algol fournissent un nouveau type de mécanisme de passage de paramètres qui fonctionne comme un préprocesseur en langage C. Dans le mécanisme de passage par nom, le nom de la procédure appelée est remplacé par son corps réel. Pass-by-name substitue textuellement les expressions d'argument dans un appel de procédure aux paramètres correspondants dans le corps de la procédure afin qu'elle puisse désormais travailler sur des paramètres réels, un peu comme le passage par référence.
La table de symboles est une structure de données importante créée et maintenue par des compilateurs afin de stocker des informations sur l'occurrence de diverses entités telles que des noms de variables, des noms de fonctions, des objets, des classes, des interfaces, etc. La table de symboles est utilisée à la fois par l'analyse et la synthèse parties d'un compilateur.
Un tableau de symboles peut servir les objectifs suivants en fonction de la langue utilisée:
Pour stocker les noms de toutes les entités sous une forme structurée à un seul endroit.
Pour vérifier si une variable a été déclarée.
Pour implémenter la vérification de type, en vérifiant que les affectations et les expressions dans le code source sont sémantiquement correctes.
Pour déterminer la portée d'un nom (résolution de la portée).
Une table de symboles est simplement une table qui peut être linéaire ou une table de hachage. Il gère une entrée pour chaque nom dans le format suivant:
<symbol name, type, attribute>
Par exemple, si une table de symboles doit stocker des informations sur la déclaration de variable suivante:
static int interest;
alors il devrait stocker l'entrée telle que:
<interest, int, static>
La clause d'attribut contient les entrées liées au nom.
la mise en oeuvre
Si un compilateur doit gérer une petite quantité de données, la table de symboles peut être implémentée sous la forme d'une liste non ordonnée, ce qui est facile à coder, mais elle ne convient que pour les petites tables. Une table de symboles peut être implémentée de l'une des manières suivantes:
- Liste linéaire (triée ou non triée)
- Arbre de recherche binaire
- Table de hachage
Parmi tous, les tables de symboles sont principalement implémentées sous forme de tables de hachage, où le symbole du code source lui-même est traité comme une clé pour la fonction de hachage et la valeur de retour est l'information sur le symbole.
Opérations
Une table de symboles, linéaire ou hachée, doit fournir les opérations suivantes.
insérer()
Cette opération est plus fréquemment utilisée par phase d'analyse, c'est-à-dire la première moitié du compilateur où les jetons sont identifiés et les noms sont stockés dans la table. Cette opération est utilisée pour ajouter des informations dans la table des symboles sur les noms uniques apparaissant dans le code source. Le format ou la structure dans lequel les noms sont stockés dépend du compilateur en main.
Un attribut d'un symbole dans le code source est l'information associée à ce symbole. Ces informations contiennent la valeur, l'état, la portée et le type du symbole. La fonction insert () prend le symbole et ses attributs comme arguments et stocke les informations dans la table des symboles.
Par exemple:
int a;
doit être traité par le compilateur comme:
insert(a, int);
Chercher()
L'opération lookup () est utilisée pour rechercher un nom dans la table des symboles afin de déterminer:
- si le symbole existe dans le tableau.
- s'il est déclaré avant d'être utilisé.
- si le nom est utilisé dans la portée.
- si le symbole est initialisé.
- si le symbole a été déclaré plusieurs fois.
Le format de la fonction lookup () varie en fonction du langage de programmation. Le format de base doit correspondre à ce qui suit:
lookup(symbol)
Cette méthode renvoie 0 (zéro) si le symbole n'existe pas dans la table des symboles. Si le symbole existe dans la table des symboles, il renvoie ses attributs stockés dans la table.
Gestion de la portée
Un compilateur gère deux types de tables de symboles: a global symbol table accessible par toutes les procédures et scope symbol tables qui sont créés pour chaque étendue du programme.
Pour déterminer la portée d'un nom, les tables de symboles sont organisées selon une structure hiérarchique, comme illustré dans l'exemple ci-dessous:
. . .
int value=10;
void pro_one()
{
int one_1;
int one_2;
{ \
int one_3; |_ inner scope 1
int one_4; |
} /
int one_5;
{ \
int one_6; |_ inner scope 2
int one_7; |
} /
}
void pro_two()
{
int two_1;
int two_2;
{ \
int two_3; |_ inner scope 3
int two_4; |
} /
int two_5;
}
. . .
Le programme ci-dessus peut être représenté dans une structure hiérarchique de tables de symboles:
La table de symboles globale contient les noms d'une variable globale (valeur int) et de deux noms de procédure, qui doivent être disponibles pour tous les nœuds enfants indiqués ci-dessus. Les noms mentionnés dans la table des symboles pro_one (et toutes ses tables enfants) ne sont pas disponibles pour les symboles pro_two et ses tables enfants.
Cette hiérarchie de structure de données de table de symboles est stockée dans l'analyseur sémantique et chaque fois qu'un nom doit être recherché dans une table de symboles, il est recherché à l'aide de l'algorithme suivant:
Tout d'abord, un symbole sera recherché dans la portée actuelle, c'est-à-dire dans la table des symboles actuelle.
si un nom est trouvé, la recherche est terminée, sinon elle sera recherchée dans la table des symboles parents jusqu'à ce que,
soit le nom est trouvé, soit la table de symboles globale a été recherchée pour le nom.
Un code source peut être directement traduit dans son code machine cible, alors pourquoi avons-nous besoin de traduire le code source en un code intermédiaire qui est ensuite traduit en son code cible? Voyons les raisons pour lesquelles nous avons besoin d'un code intermédiaire.
Si un compilateur traduit le langage source vers son langage machine cible sans avoir la possibilité de générer du code intermédiaire, alors pour chaque nouvelle machine, un compilateur natif complet est requis.
Le code intermédiaire élimine le besoin d'un nouveau compilateur complet pour chaque machine unique en gardant la partie d'analyse identique pour tous les compilateurs.
La deuxième partie du compilateur, la synthèse, est modifiée en fonction de la machine cible.
Il devient plus facile d'appliquer les modifications du code source pour améliorer les performances du code en appliquant des techniques d'optimisation du code sur le code intermédiaire.
Représentation intermédiaire
Les codes intermédiaires peuvent être représentés de diverses manières et ils ont leurs propres avantages.
High Level IR- La représentation du code intermédiaire de haut niveau est très proche du langage source lui-même. Ils peuvent être facilement générés à partir du code source et nous pouvons facilement appliquer des modifications de code pour améliorer les performances. Mais pour l'optimisation de la machine cible, c'est moins préféré.
Low Level IR - Celui-ci est proche de la machine cible, ce qui le rend approprié pour l'allocation de registre et de mémoire, la sélection de jeu d'instructions, etc. C'est bon pour les optimisations dépendant de la machine.
Le code intermédiaire peut être soit spécifique à la langue (par exemple, Byte Code pour Java), soit indépendant du langage (code à trois adresses).
Code à trois adresses
Le générateur de code intermédiaire reçoit l'entrée de sa phase prédécesseur, l'analyseur sémantique, sous la forme d'un arbre de syntaxe annoté. Cet arbre de syntaxe peut alors être converti en une représentation linéaire, par exemple, la notation postfixe. Le code intermédiaire a tendance à être un code indépendant de la machine. Par conséquent, le générateur de code suppose d'avoir un nombre illimité de stockage mémoire (registre) pour générer du code.
Par exemple:
a = b + c * d;
Le générateur de code intermédiaire essaiera de diviser cette expression en sous-expressions puis de générer le code correspondant.
r1 = c * d;
r2 = b + r1;
a = r2
r utilisé comme registres dans le programme cible.
Un code à trois adresses a au plus trois emplacements d'adresses pour calculer l'expression. Un code à trois adresses peut être représenté sous deux formes: quadruples et triples.
Quadruples
Chaque instruction dans la présentation en quadruples est divisée en quatre champs: opérateur, arg1, arg2 et résultat. L'exemple ci-dessus est représenté ci-dessous au format quadruples:
Op | arg 1 | arg 2 | résultat |
* | c | ré | r1 |
+ | b | r1 | r2 |
+ | r2 | r1 | r3 |
= | r3 | une |
Triples
Chaque instruction dans la présentation en triplets a trois champs: op, arg1 et arg2. Les résultats des sous-expressions respectives sont indiqués par la position de l'expression. Les triplets représentent la similitude avec le DAG et l'arbre syntaxique. Ils sont équivalents à DAG lorsqu'ils représentent des expressions.
Op | arg 1 | arg 2 |
* | c | ré |
+ | b | (0) |
+ | (1) | (0) |
= | (2) |
Les triplets sont confrontés au problème de l'immobilité du code lors de l'optimisation, car les résultats sont positionnels et la modification de l'ordre ou de la position d'une expression peut entraîner des problèmes.
Triples indirects
Cette représentation est une amélioration par rapport à la représentation en triplets. Il utilise des pointeurs au lieu de la position pour stocker les résultats. Cela permet aux optimiseurs de repositionner librement la sous-expression pour produire un code optimisé.
Déclarations
Une variable ou une procédure doit être déclarée avant de pouvoir être utilisée. La déclaration implique l'allocation d'espace en mémoire et l'entrée du type et du nom dans la table des symboles. Un programme peut être codé et conçu en gardant à l'esprit la structure de la machine cible, mais il n'est pas toujours possible de convertir avec précision un code source dans sa langue cible.
En prenant l'ensemble du programme comme un ensemble de procédures et de sous-procédures, il devient possible de déclarer tous les noms locaux à la procédure. L'allocation de mémoire se fait de manière consécutive et les noms sont alloués à la mémoire dans l'ordre où ils sont déclarés dans le programme. Nous utilisons la variable offset et la mettons à zéro {offset = 0} qui indique l'adresse de base.
Le langage de programmation source et l'architecture de la machine cible peuvent varier dans la façon dont les noms sont stockés, donc un adressage relatif est utilisé. Alors que la mémoire est allouée au premier nom à partir de l'emplacement de mémoire 0 {offset = 0}, le nom suivant déclaré plus tard doit être alloué en mémoire à côté du premier.
Example:
Nous prenons l'exemple du langage de programmation C où une variable entière se voit attribuer 2 octets de mémoire et une variable float se voit attribuer 4 octets de mémoire.
int a;
float b;
Allocation process:
{offset = 0}
int a;
id.type = int
id.width = 2
offset = offset + id.width
{offset = 2}
float b;
id.type = float
id.width = 4
offset = offset + id.width
{offset = 6}
Pour saisir ce détail dans une table de symboles, une entrée de procédure peut être utilisée. Cette méthode peut avoir la structure suivante:
enter(name, type, offset)
Cette procédure doit créer une entrée dans la table des symboles, pour le nom de la variable , dont le type est défini sur type et le décalage d' adresse relatif dans sa zone de données.
La génération de code peut être considérée comme la phase finale de la compilation. Grâce à la génération de code postal, le processus d'optimisation peut être appliqué au code, mais cela peut être considéré comme une partie de la phase de génération de code elle-même. Le code généré par le compilateur est un code objet d'un langage de programmation de niveau inférieur, par exemple, le langage d'assemblage. Nous avons vu que le code source écrit dans un langage de niveau supérieur est transformé en un langage de niveau inférieur qui se traduit par un code objet de niveau inférieur, qui doit avoir les propriétés minimales suivantes:
- Il doit porter la signification exacte du code source.
- Il doit être efficace en termes d'utilisation du processeur et de gestion de la mémoire.
Nous allons maintenant voir comment le code intermédiaire est transformé en code objet cible (code assembleur, dans ce cas).
Graphe acyclique dirigé
Directed Acyclic Graph (DAG) est un outil qui décrit la structure des blocs de base, aide à voir le flux de valeurs circulant entre les blocs de base et offre également une optimisation. DAG permet une transformation facile sur des blocs de base. DAG peut être compris ici:
Les nœuds feuilles représentent des identificateurs, des noms ou des constantes.
Les nœuds intérieurs représentent les opérateurs.
Les nœuds intérieurs représentent également les résultats d'expressions ou les identificateurs / nom où les valeurs doivent être stockées ou attribuées.
Example:
t0 = a + b
t1 = t0 + c
d = t0 + t1
[t 0 = a + b] |
[t 1 = t 0 + c] |
[d = t 0 + t 1 ] |
Optimisation du judas
Cette technique d'optimisation fonctionne localement sur le code source pour le transformer en un code optimisé. Par localement, nous entendons une petite partie du bloc de code disponible. Ces méthodes peuvent être appliquées sur des codes intermédiaires ainsi que sur des codes cibles. Un groupe d'instructions est analysé et vérifié pour l'optimisation possible suivante:
Élimination des instructions redondantes
Au niveau du code source, les opérations suivantes peuvent être effectuées par l'utilisateur:
|
|
|
|
Au niveau de la compilation, le compilateur recherche des instructions de nature redondante. Le chargement et le stockage multiples d'instructions peuvent avoir la même signification même si certaines d'entre elles sont supprimées. Par exemple:
- MOV x, R0
- MOV R0, R1
Nous pouvons supprimer la première instruction et réécrire la phrase comme suit:
MOV x, R1
Code inaccessible
Le code inaccessible est une partie du code du programme qui n'est jamais accédée en raison des constructions de programmation. Les programmeurs peuvent avoir accidentellement écrit un morceau de code qui ne peut jamais être atteint.
Example:
void add_ten(int x)
{
return x + 10;
printf(“value of x is %d”, x);
}
Dans ce segment de code, le printf l'instruction ne sera jamais exécutée lorsque le contrôle du programme revient avant de pouvoir s'exécuter, d'où printf Peut être enlevé.
Optimisation des flux de contrôle
Il existe des cas dans un code où le contrôle du programme saute d'avant en arrière sans effectuer aucune tâche significative. Ces sauts peuvent être supprimés. Considérez le morceau de code suivant:
...
MOV R1, R2
GOTO L1
...
L1 : GOTO L2
L2 : INC R1
Dans ce code, l'étiquette L1 peut être supprimée lorsqu'elle passe le contrôle à L2. Ainsi, au lieu de sauter à L1 puis à L2, la commande peut directement atteindre L2, comme indiqué ci-dessous:
...
MOV R1, R2
GOTO L2
...
L2 : INC R1
Simplification de l'expression algébrique
Il y a des occasions où les expressions algébriques peuvent être simplifiées. Par exemple, l'expressiona = a + 0 peut être remplacé par a lui-même et l'expression a = a + 1 peut simplement être remplacée par INC a.
Réduction de la force
Il y a des opérations qui consomment plus de temps et d'espace. Leur «force» peut être réduite en les remplaçant par d'autres opérations qui consomment moins de temps et d'espace, mais produisent le même résultat.
Par exemple, x * 2 peut être remplacé par x << 1, qui n'implique qu'un seul décalage à gauche. Bien que la sortie d'un * a et d'un 2 soit la même, un 2 est beaucoup plus efficace à implémenter.
Accéder aux instructions de la machine
La machine cible peut déployer des instructions plus sophistiquées, qui peuvent avoir la capacité d'effectuer des opérations spécifiques de manière beaucoup plus efficace. Si le code cible peut accueillir ces instructions directement, cela améliorera non seulement la qualité du code, mais produira également des résultats plus efficaces.
Générateur de code
Un générateur de code doit comprendre l'environnement d'exécution de la machine cible et son jeu d'instructions. Le générateur de code doit prendre en compte les éléments suivants pour générer le code:
Target language: Le générateur de code doit être conscient de la nature du langage cible pour lequel le code doit être transformé. Ce langage peut faciliter certaines instructions spécifiques à la machine pour aider le compilateur à générer le code d'une manière plus pratique. La machine cible peut avoir une architecture de processeur CISC ou RISC.
IR Type: La représentation intermédiaire a différentes formes. Il peut s'agir d'une structure AST (Abstract Syntax Tree), d'une notation polonaise inversée ou d'un code à 3 adresses.
Selection of instruction: Le générateur de code prend la représentation intermédiaire comme entrée et la convertit (mappe) dans le jeu d'instructions de la machine cible. Une représentation peut avoir de nombreuses façons (instructions) de la convertir, il incombe donc au générateur de code de choisir judicieusement les instructions appropriées.
Register allocation: Un programme a un certain nombre de valeurs à conserver pendant l'exécution. L'architecture de la machine cible peut ne pas permettre à toutes les valeurs d'être conservées dans la mémoire ou les registres du processeur. Le générateur de code décide des valeurs à conserver dans les registres. En outre, il décide des registres à utiliser pour conserver ces valeurs.
Ordering of instructions: Enfin, le générateur de code décide de l'ordre dans lequel l'instruction sera exécutée. Il crée des horaires pour les instructions pour les exécuter.
Descripteurs
Le générateur de code doit suivre à la fois les registres (pour la disponibilité) et les adresses (emplacement des valeurs) tout en générant le code. Pour les deux, les deux descripteurs suivants sont utilisés:
Register descriptor: Le descripteur de registre est utilisé pour informer le générateur de code de la disponibilité des registres. Le descripteur de registre garde une trace des valeurs stockées dans chaque registre. Chaque fois qu'un nouveau registre est requis pendant la génération de code, ce descripteur est consulté pour la disponibilité du registre.
Address descriptor: Les valeurs des noms (identificateurs) utilisés dans le programme peuvent être stockées à différents endroits lors de l'exécution. Les descripteurs d'adresses sont utilisés pour garder une trace des emplacements de mémoire où les valeurs des identificateurs sont stockées. Ces emplacements peuvent inclure des registres CPU, des tas, des piles, de la mémoire ou une combinaison des emplacements mentionnés.
Le générateur de code maintient à la fois le descripteur mis à jour en temps réel. Pour une instruction de chargement, LD R1, x, le générateur de code:
- met à jour le descripteur de registre R1 qui a la valeur x et
- met à jour le descripteur d'adresse (x) pour montrer qu'une instance de x est dans R1.
Génération de code
Les blocs de base se composent d'une séquence d'instructions à trois adresses. Le générateur de code prend cette séquence d'instructions comme entrée.
Note: Si la valeur d'un nom est trouvée à plusieurs endroits (registre, cache ou mémoire), la valeur du registre sera préférée au cache et à la mémoire principale. De même, la valeur du cache sera préférée à la mémoire principale. La mémoire principale n'a guère de préférence.
getReg: Le générateur de code utilise la fonction getReg pour déterminer l'état des registres disponibles et l'emplacement des valeurs de nom. getReg fonctionne comme suit:
Si la variable Y est déjà dans le registre R, elle utilise ce registre.
Sinon, si un registre R est disponible, il utilise ce registre.
Sinon, si les deux options ci-dessus ne sont pas possibles, il choisit un registre qui nécessite un nombre minimal d'instructions de chargement et de stockage.
Pour une instruction x = y OP z, le générateur de code peut effectuer les actions suivantes. Supposons que L est l'emplacement (de préférence le registre) où la sortie de y OP z doit être sauvegardée:
Appelez la fonction getReg, pour décider de l'emplacement de L.
Déterminez l'emplacement actuel (registre ou mémoire) de y en consultant le descripteur d'adresse de y. Siy n'est pas actuellement inscrit L, puis générez l'instruction suivante pour copier la valeur de y à L:
MOV y ', L
où y’ représente la valeur copiée de y.
Déterminez l'emplacement actuel de z en utilisant la même méthode utilisée à l'étape 2 pour y et générez l'instruction suivante:
OP z ', L
où z’ représente la valeur copiée de z.
Maintenant L contient la valeur de y OP z, qui est destinée à être affectée à x. Donc, si L est un registre, mettez à jour son descripteur pour indiquer qu'il contient la valeur dex. Mettre à jour le descripteur dex pour indiquer qu'il est stocké à l'emplacement L.
Si y et z n'ont plus aucune utilité, ils peuvent être restitués au système.
D'autres constructions de code comme les boucles et les instructions conditionnelles sont transformées en langage d'assemblage de manière d'assemblage général.
L'optimisation est une technique de transformation de programme, qui tente d'améliorer le code en le faisant consommer moins de ressources (c.-à-d. CPU, mémoire) et fournir une vitesse élevée.
Dans l'optimisation, les constructions de programmation générale de haut niveau sont remplacées par des codes de programmation de bas niveau très efficaces. Un processus d'optimisation de code doit suivre les trois règles ci-dessous:
Le code de sortie ne doit en aucun cas changer la signification du programme.
L'optimisation devrait augmenter la vitesse du programme et si possible, le programme devrait exiger moins de ressources.
L'optimisation doit elle-même être rapide et ne doit pas retarder le processus global de compilation.
Des efforts pour un code optimisé peuvent être faits à différents niveaux de compilation du processus.
Au début, les utilisateurs peuvent modifier / réorganiser le code ou utiliser de meilleurs algorithmes pour écrire le code.
Après avoir généré du code intermédiaire, le compilateur peut modifier le code intermédiaire par des calculs d'adresse et des boucles améliorées.
Lors de la production du code machine cible, le compilateur peut utiliser la hiérarchie de la mémoire et les registres du processeur.
L'optimisation peut être classée en deux grandes catégories: indépendante de la machine et dépendante de la machine.
Optimisation indépendante de la machine
Dans cette optimisation, le compilateur prend le code intermédiaire et transforme une partie du code qui n'implique aucun registre CPU et / ou emplacements de mémoire absolus. Par exemple:
do
{
item = 10;
value = value + item;
}while(value<100);
Ce code implique une affectation répétée de l'élément identifiant, qui si nous mettons de cette façon:
Item = 10;
do
{
value = value + item;
} while(value<100);
devrait non seulement enregistrer les cycles du processeur, mais peut être utilisé sur n'importe quel processeur.
Optimisation en fonction de la machine
L'optimisation dépendante de la machine est effectuée une fois que le code cible a été généré et lorsque le code est transformé en fonction de l'architecture de la machine cible. Il implique des registres CPU et peut avoir des références de mémoire absolues plutôt que des références relatives. Les optimiseurs dépendant de la machine s'efforcent de tirer le meilleur parti de la hiérarchie de la mémoire.
Blocs de base
Les codes sources ont généralement un certain nombre d'instructions, qui sont toujours exécutées en séquence et sont considérées comme les blocs de base du code. Ces blocs de base n'ont pas d'instructions de saut parmi eux, c'est-à-dire que lorsque la première instruction est exécutée, toutes les instructions d'un même bloc de base seront exécutées dans leur séquence d'apparition sans perdre le contrôle de flux du programme.
Un programme peut avoir diverses constructions en tant que blocs de base, comme des instructions conditionnelles IF-THEN-ELSE, SWITCH-CASE et des boucles telles que DO-WHILE, FOR et REPEAT-UNTIL, etc.
Identification de bloc de base
Nous pouvons utiliser l'algorithme suivant pour trouver les blocs de base dans un programme:
Rechercher les instructions d'en-tête de tous les blocs de base à partir desquels un bloc de base commence:
- Premier énoncé d'un programme.
- Les instructions qui sont la cible de n'importe quelle branche (conditionnelle / inconditionnelle).
- Déclarations qui suivent toute instruction de branche.
Les instructions d'en-tête et les instructions qui les suivent forment un bloc de base.
Un bloc de base n'inclut aucune instruction d'en-tête d'aucun autre bloc de base.
Les blocs de base sont des concepts importants du point de vue de la génération de code et de l'optimisation.
Les blocs de base jouent un rôle important dans l'identification des variables, qui sont utilisées plus d'une fois dans un seul bloc de base. Si une variable est utilisée plus d'une fois, la mémoire de registre allouée à cette variable n'a pas besoin d'être vidée à moins que le bloc ne termine l'exécution.
Graphique de flux de contrôle
Les blocs de base d'un programme peuvent être représentés au moyen de graphiques de flux de contrôle. Un graphique de flux de contrôle décrit la manière dont le contrôle du programme est transmis entre les blocs. C'est un outil utile qui aide à l'optimisation en aidant à localiser les boucles indésirables dans le programme.
Optimisation des boucles
La plupart des programmes s'exécutent en boucle dans le système. Il devient nécessaire d'optimiser les boucles pour économiser les cycles CPU et la mémoire. Les boucles peuvent être optimisées par les techniques suivantes:
Invariant code: Un fragment de code qui réside dans la boucle et calcule la même valeur à chaque itération est appelé un code invariant de boucle. Ce code peut être déplacé hors de la boucle en l'enregistrant pour être calculé une seule fois, plutôt qu'à chaque itération.
Induction analysis : Une variable est appelée variable d'induction si sa valeur est modifiée dans la boucle par une valeur invariante de boucle.
Strength reduction: Il existe des expressions qui consomment plus de cycles CPU, de temps et de mémoire. Ces expressions doivent être remplacées par des expressions moins chères sans compromettre la sortie de l'expression. Par exemple, la multiplication (x * 2) est coûteuse en termes de cycles CPU que (x << 1) et donne le même résultat.
Élimination du code mort
Le code mort est une ou plusieurs instructions de code, qui sont:
- Soit jamais exécuté, soit inaccessible,
- Ou s'ils sont exécutés, leur sortie n'est jamais utilisée.
Ainsi, le code mort ne joue aucun rôle dans aucune opération du programme et peut donc être simplement éliminé.
Code partiellement mort
Il existe des instructions de code dont les valeurs calculées ne sont utilisées que dans certaines circonstances, c'est-à-dire que parfois les valeurs sont utilisées et parfois elles ne le sont pas. Ces codes sont connus sous le nom de code mort partiel.
Le graphe de flux de contrôle ci-dessus représente un morceau de programme où la variable «a» est utilisée pour affecter la sortie de l'expression «x * y». Supposons que la valeur assignée à 'a' ne soit jamais utilisée à l'intérieur de la boucle. Immédiatement après que le contrôle ait quitté la boucle, 'a' se voit attribuer la valeur de la variable 'z', qui sera utilisée plus tard dans le programme. Nous concluons ici que le code d'attribution de «a» n'est jamais utilisé nulle part, il est donc éligible pour être éliminé.
De même, l'image ci-dessus montre que l'instruction conditionnelle est toujours fausse, ce qui implique que le code, écrit en cas vrai, ne sera jamais exécuté, par conséquent il peut être supprimé.
Redondance partielle
Les expressions redondantes sont calculées plus d'une fois dans un chemin parallèle, sans aucune modification des opérandes, alors que les expressions partiellement redondantes sont calculées plus d'une fois dans un chemin, sans aucun changement d'opérandes. Par exemple,
[expression redondante] |
[expression partiellement redondante] |
Le code invariant en boucle est partiellement redondant et peut être éliminé en utilisant une technique de mouvement de code.
Un autre exemple de code partiellement redondant peut être:
If (condition)
{
a = y OP z;
}
else
{
...
}
c = y OP z;
Nous supposons que les valeurs des opérandes (y et z) ne sont pas modifiés depuis l'affectation de la variable a à variable c. Ici, si l'instruction de condition est vraie, alors y OP z est calculé deux fois, sinon une fois. Le mouvement de code peut être utilisé pour éliminer cette redondance, comme indiqué ci-dessous:
If (condition)
{
...
tmp = y OP z;
a = tmp;
...
}
else
{
...
tmp = y OP z;
}
c = tmp;
Ici, si la condition est vraie ou fausse; y OP z ne doit être calculé qu'une seule fois.