This repository has been archived by the owner on May 12, 2022. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathpassePlacementRat.ml
104 lines (97 loc) · 4.1 KB
/
passePlacementRat.ml
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
module PassePlacementRat :
Passe.Passe
with type t1 = Ast.AstType.programme
and type t2 = Ast.AstPlacement.programme = struct
open Tds
open Exceptions
open Ast
open AstPlacement
open Type
type t1 = Ast.AstType.programme
type t2 = Ast.AstPlacement.programme
(* analyse_placement_struct : string -> info_ast -> unit *)
(* Param reg : registre de la variable *)
(* Param ia : info_ast à modifier *)
(* Check si c'est une structure et place ses attributs *relativement* à celle-ci *)
let rec analyse_placement_struct reg ia =
let analyse_placement_champ dep ia =
let t = getTaille (get_type ia) in
modifier_adresse_info dep reg ia;
analyse_placement_struct reg ia;
dep + t
in
match info_ast_to_info ia with
| InfoStruct (_, _, _, _, attrs) ->
let _ = List.fold_left analyse_placement_champ 0 attrs in
()
| _ -> ()
(* analyse_placement_instruction : string -> int -> AstType.instruction -> int *)
(* Paramètre reg: le registre mémoire *)
(* Paramètre dep: la profondeur actuelle dans le registre *)
(* Paramètre i: l'instruction à analyser *)
(* Réserve dans le registre la place nécéssaire aux variables déclarée
et analyse les bloc des structures de contrôle *)
(* Renvoie la nouvelle position dans le registre *)
let rec analyse_placement_instruction reg dep i =
(* On récupère l'incrément de profondeur lié à l'instruction *)
let inc =
match i with
(* Déclaration: il faut réserver la place de la nouvelle variable *)
| AstType.Declaration (ia, _) ->
(* Place la variable à l'adresse dep du registre reg *)
modifier_adresse_info dep reg ia;
analyse_placement_struct reg ia;
(* renvoie la taille du type déclaré *)
getTaille (get_type ia)
| AstType.Conditionnelle (_, bt, be) ->
(* On analyse les deux blocs, la conditionnelle en tant que telle
ne consomme aucun espace mémoire à la fin *)
analyse_placement_bloc reg dep bt;
analyse_placement_bloc reg dep be;
0
| AstType.TantQue (_, b) ->
(* De même avec le tant que *)
analyse_placement_bloc reg dep b;
0
| _ -> 0
in
(* nouvelle profondeur (pour la prochaine déclaration) *)
dep + inc
(* analyse_placement_params: int -> info_ast list -> unit = <fun> *)
(* Place les paramètres en mémoire par effet de bord.
Params:
- reg: string = le nom du registre mémoire dans lequel stocker les variables
- dep: int = le déplacement de base dans notre registre
- li: instruction list = la liste des instructions dont on va gérer la placement mémoire
Retour: unit *)
and analyse_placement_bloc reg dep li =
let _ = List.fold_left (analyse_placement_instruction reg) dep li in
()
(* analyse_placement_fonction: AstType.Fonction -> AstPlacement.Fonction = <fun> *)
(* Par effet de bord, gère le placement mémoire des paramètres et des variables du bloc de la fonction *)
let rec analyse_placement_fonction (AstType.Fonction (ia, liap, li)) =
analyse_placement_params 0 (List.rev liap);
analyse_placement_bloc "LB" 3 li;
Fonction (ia, liap, li)
(* analyse_placement_params: int -> info_ast list -> unit = <fun> *)
(* Place les paramètres en mémoire par effet de bord.
Params:
- dep: int = le déplacement de base de notre paramètre
- lp: info_ast list = la liste des paramètres à placer dans la mémoire
Retour: unit *)
and analyse_placement_params dep liap =
let rec analyse_placement_param dep ia =
let t = getTaille (get_type ia) in
modifier_adresse_info (dep - t) "LB" ia;
analyse_placement_struct "LB" ia;
dep - t
in
let _ = List.fold_left analyse_placement_param dep liap in
()
(* analyser : AstType.Programme -> AstPlacement.Programme *)
(* Paramètre : le programme à analyser *)
let analyser (AstType.Programme (fonctions, prog)) =
let nf = List.map analyse_placement_fonction fonctions in
analyse_placement_bloc "SB" 0 prog;
Programme (nf, prog)
end