Fin du readme
This commit is contained in:
parent
45723682bf
commit
d8df43acf3
49
README.md
49
README.md
|
@ -12,7 +12,7 @@ geometry:
|
||||||
|
|
||||||
# Analyse lexicale
|
# Analyse lexicale
|
||||||
|
|
||||||
Le fichier `example.l` peut se compiler en un analyseur lexical `example` en appelant simplement la commande `make`. Il permet à partir d'une entrée textuelle (un fichier, le TTY ou l'entrée standard suivie d'un Ctrl+D) de filter les nombres et de les afficher, sous forme décimale ou hexadécimale. De plus, les mots clés `if`, `then` et `else` sont réécrits en majsucule. Tout le reste est ignoré.
|
Le fichier `example.l` peut se compiler en un analyseur lexical `example` en appelant simplement la commande `make example`. Il permet à partir d'une entrée textuelle (un fichier, le TTY ou l'entrée standard suivie d'un Ctrl+D) de filter les nombres et de les afficher, sous forme décimale ou hexadécimale. De plus, les mots clés `if`, `then` et `else` sont réécrits en majsucule. Tout le reste est ignoré.
|
||||||
|
|
||||||
Un entier `n` affichera sur la sortie `int(n)`, un entier `0xh` écrit en hexadécimal affichera dans la sortie `hex(n)` où `n` est la représentation décimale de `0xh`.
|
Un entier `n` affichera sur la sortie `int(n)`, un entier `0xh` écrit en hexadécimal affichera dans la sortie `hex(n)` où `n` est la représentation décimale de `0xh`.
|
||||||
|
|
||||||
|
@ -34,6 +34,51 @@ else
|
||||||
Le programme va renvoyer la sortie suivante :
|
Le programme va renvoyer la sortie suivante :
|
||||||
|
|
||||||
```
|
```
|
||||||
int(1)IFTHENint(1)int(20)int(20)int(100)int(3)ELSEhex(16777215)int(10)int(12)int(42)int(13)int(7832)
|
int(1)IFTHENint(1)int(20)int(20)int(100)int(3)ELSEhex(16777215)int(10)int(12)int(42)int(14)int(7832)
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
# Analyse syntaxique
|
||||||
|
|
||||||
|
Le programme `lang`, après compilation en appelant successivement `make langlex.c` et `make lang`, se comporte comme un interpréteur d'un langage de programmation fait à la main. L'extension attribuée est le `*.my`, même si cela n'est pas obligatoire.
|
||||||
|
|
||||||
|
Un programme est composé de trois parties :
|
||||||
|
|
||||||
|
* La déclaration des variables booléennes (faculatif). Cela se fait en écrivant `bool` suivi de la liste des variables booléennes à déclarer séparées par des virgules. La déclaration doit se finir par un point-virgule. Par exemple, on notera `bool x, y, z;` pour déclarer les variables booléenes `x`, `y` et `z`.
|
||||||
|
|
||||||
|
* De la même manière, on notera `int a, b, c;` pour déclarer les variables entières `a`, `b` et `c` (facultatif). La déclaration des variables entières doit se faire après la déclaration des variables booléennes s'il y en a.
|
||||||
|
|
||||||
|
* Enfin, le code. Le code est une succession d'instructions séparées par des point-virgules. Il existe 4 types d'instructions :
|
||||||
|
* L'affectation, de la forme `x := expr` où `x` est une variable déclarée et `expr` une expression du même type que `x`.
|
||||||
|
* Un test conditionnel, de la forme `if cond then stmt1 else stmt2` où `cond` est une expression booléenne et `stmt1` et `stmt2` deux blocs d'instructions. L'interprétation est naturelle : si la condition `cond` vaut `true`, on exécute le bloc `stmt1`, sinon on évalue le bloc `stmt2`. La partie `else stmt2` est facultative. Le problème du `dangling else` est résolu de la même manière qu'en C : le premier `else` porte sur le premier `if`.
|
||||||
|
* Une boucle de la forme `while cond do stmt od` où `cond` est une expression booléenne et `stmt` une suite d'instructions. L'expression `cond` sera évaluée à chaque passage de boucle, et on reste dans la boucle tant que que la condition vaut `true`.
|
||||||
|
* Un affichage des variables. Le code doit être de la forme `print list` où `list` est la liste des variables à afficher, séparées par des virgules. La fonction fonctionne à la fois avec des variables booléennes et des variables entières.
|
||||||
|
|
||||||
|
Une expression peut être de différentes formes. Chaque type d'expression est associée à une "classe" : booléen ou entier, ou les deux.
|
||||||
|
|
||||||
|
* La valeur d'une variable (du type de la variable)
|
||||||
|
* Un ou exclusif de deux expressions booléennes (renvoie un booléen)
|
||||||
|
* La disjonction de deux expressions booléennes (renvoie un booléen)
|
||||||
|
* La conjection de deux expressions booléennes (renvoie un booléen)
|
||||||
|
* L'équivalence deux expressions booléennes (renvoie un booléen)
|
||||||
|
* Une négation d'une expression booléenne (renvoie un booléen)
|
||||||
|
* `true` (booléen pour dire vrai)
|
||||||
|
* `false` (booléen pour dire faux)
|
||||||
|
* Une expression entre parenthèse (du type de ce qui est entre parenthèses)
|
||||||
|
* La somme de deux expressions entières (renvoie un entier)
|
||||||
|
* Le produit de deux expressions entières (renvoie un entier)
|
||||||
|
* La comparaison par inégalité stricte entre deux expression entières (renvoie un booléen)
|
||||||
|
* L'égalité entre deux expressions entières (renvoie un booléen)
|
||||||
|
|
||||||
|
Les priorités opérations sont les mêmes qu'en C :
|
||||||
|
|
||||||
|
1. La négation booléenne
|
||||||
|
2. La mutliplication entière
|
||||||
|
3. L'addition entière
|
||||||
|
4. L'inégalité stricte entre entiers
|
||||||
|
5. L'égalité entière
|
||||||
|
6. L'équivalence booléenne
|
||||||
|
7. La conjonction
|
||||||
|
8. La disjonction (éventuellement exclusive)
|
||||||
|
|
||||||
|
Lors de l'évaluation des expressions, les types sont correctement vérifiés.
|
||||||
|
|
BIN
README.pdf
BIN
README.pdf
Binary file not shown.
|
@ -108,6 +108,8 @@ expr* make_expr (int type, int class, var *var, expr *left, expr *right)
|
||||||
expr *e = malloc(sizeof(expr));
|
expr *e = malloc(sizeof(expr));
|
||||||
e->type = type;
|
e->type = type;
|
||||||
e->class = class;
|
e->class = class;
|
||||||
|
if (var != NULL)
|
||||||
|
e->class = var->class;
|
||||||
e->var = var;
|
e->var = var;
|
||||||
e->left = left;
|
e->left = left;
|
||||||
e->right = right;
|
e->right = right;
|
||||||
|
@ -230,28 +232,28 @@ int eval (expr *e)
|
||||||
case FALSE: return 0;
|
case FALSE: return 0;
|
||||||
case XOR: {
|
case XOR: {
|
||||||
if (e->left->class != 0 || e->right->class != 0) {
|
if (e->left->class != 0 || e->right->class != 0) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("A xor expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) ^ eval(e->right);
|
return eval(e->left) ^ eval(e->right);
|
||||||
}
|
}
|
||||||
case OR: {
|
case OR: {
|
||||||
if (e->left->class != 0 || e->right != NULL && e->right->class != 0) {
|
if (e->left->class != 0 || e->right != NULL && e->right->class != 0) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("An or expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) || eval(e->right);
|
return eval(e->left) || eval(e->right);
|
||||||
}
|
}
|
||||||
case AND: {
|
case AND: {
|
||||||
if (e->left->class != 0 || e->right->class != 0) {
|
if (e->left->class != 0 || e->right->class != 0) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("An and expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) && eval(e->right);
|
return eval(e->left) && eval(e->right);
|
||||||
}
|
}
|
||||||
case EQUIV: {
|
case EQUIV: {
|
||||||
if (e->left->class != 0 || e->right->class != 0) {
|
if (e->left->class != 0 || e->right->class != 0) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("An equivalence expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
int right = eval(e->right);
|
int right = eval(e->right);
|
||||||
|
@ -261,28 +263,28 @@ int eval (expr *e)
|
||||||
}
|
}
|
||||||
case NOT: {
|
case NOT: {
|
||||||
if (e->left->class != 0) {
|
if (e->left->class != 0) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("A not expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return !eval(e->left);
|
return !eval(e->left);
|
||||||
}
|
}
|
||||||
case PLUS: {
|
case PLUS: {
|
||||||
if (e->left->class != 1 || e->right->class != 1) {
|
if (e->left->class != 1 || e->right->class != 1) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("A plus expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) + eval(e->right);
|
return eval(e->left) + eval(e->right);
|
||||||
}
|
}
|
||||||
case TIMES: {
|
case TIMES: {
|
||||||
if (e->left->class != 1 || e->right->class != 1) {
|
if (e->left->class != 1 || e->right->class != 1) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("A times expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) * eval(e->right);
|
return eval(e->left) * eval(e->right);
|
||||||
}
|
}
|
||||||
case EQUALS: {
|
case EQUALS: {
|
||||||
if (e->left->class != 1 || e->right->class != 1) {
|
if (e->left->class != 1 || e->right->class != 1) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("An equality expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -291,7 +293,7 @@ int eval (expr *e)
|
||||||
}
|
}
|
||||||
case LT: {
|
case LT: {
|
||||||
if (e->left->class != 1 || e->right->class != 1) {
|
if (e->left->class != 1 || e->right->class != 1) {
|
||||||
yyerror("An expression has wrong types");
|
yyerror("A comparison expression has wrong types");
|
||||||
exit(-1);
|
exit(-1);
|
||||||
}
|
}
|
||||||
return eval(e->left) < eval(e->right) ? 1 : 0;
|
return eval(e->left) < eval(e->right) ? 1 : 0;
|
||||||
|
@ -332,15 +334,26 @@ void execute (stmt *s)
|
||||||
execute(s->left);
|
execute(s->left);
|
||||||
execute(s->right);
|
execute(s->right);
|
||||||
break;
|
break;
|
||||||
case WHILE:
|
case WHILE: {
|
||||||
|
if (s->expr->class != 0 && s->expr->class != 2) {
|
||||||
|
yyerror("A while condition must be a boolean expression");
|
||||||
|
exit(-1);
|
||||||
|
}
|
||||||
|
|
||||||
while (eval(s->expr)) execute(s->left);
|
while (eval(s->expr)) execute(s->left);
|
||||||
break;
|
break;
|
||||||
case IF:
|
}
|
||||||
|
case IF: {
|
||||||
|
if (s->expr->class != 0 && s->expr->class != 2) {
|
||||||
|
yyerror("An if condition must be a boolean expression");
|
||||||
|
exit(-1);
|
||||||
|
}
|
||||||
if (eval(s->expr))
|
if (eval(s->expr))
|
||||||
execute(s->left);
|
execute(s->left);
|
||||||
else if (s->right != NULL)
|
else if (s->right != NULL)
|
||||||
execute(s->right);
|
execute(s->right);
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
case PRINT:
|
case PRINT:
|
||||||
print_vars(s->list);
|
print_vars(s->list);
|
||||||
puts("");
|
puts("");
|
||||||
|
|
Loading…
Reference in New Issue