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
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
|
-- Statements synthesis.
-- Copyright (C) 2017 Tristan Gingold
--
-- This file is part of GHDL.
--
-- This program is free software: you can redistribute it and/or modify
-- it under the terms of the GNU General Public License as published by
-- the Free Software Foundation, either version 2 of the License, or
-- (at your option) any later version.
--
-- This program is distributed in the hope that it will be useful,
-- but WITHOUT ANY WARRANTY; without even the implied warranty of
-- MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
-- GNU General Public License for more details.
--
-- You should have received a copy of the GNU General Public License
-- along with this program. If not, see <gnu.org/licenses>.
with Types; use Types;
with Vhdl.Nodes; use Vhdl.Nodes;
with Elab.Vhdl_Context; use Elab.Vhdl_Context;
with Elab.Vhdl_Objtypes; use Elab.Vhdl_Objtypes;
with Elab.Vhdl_Values; use Elab.Vhdl_Values;
with Netlists; use Netlists;
with Synth.Vhdl_Environment; use Synth.Vhdl_Environment.Env;
package Synth.Vhdl_Stmts is
type Assertion_Report_Handler_Acc is access procedure
(Inst : Synth_Instance_Acc;
Stmt : Node;
Severity : Natural;
Msg : Valtyp);
-- Procedure to call for report/assertion message.
Assertion_Report_Handler : Assertion_Report_Handler_Acc;
-- Create a new Synth_Instance for calling subprogram IMP/BOD.
function Synth_Subprogram_Call_Instance (Inst : Synth_Instance_Acc;
Imp : Node;
Bod : Node)
return Synth_Instance_Acc;
function Synth_Protected_Call_Instance (Inst : Synth_Instance_Acc;
Obj : Node;
Imp : Node;
Bod : Node)
return Synth_Instance_Acc;
procedure Synth_Subprogram_Associations (Subprg_Inst : Synth_Instance_Acc;
Caller_Inst : Synth_Instance_Acc;
Inter_Chain : Node;
Assoc_Chain : Node;
Call_Loc : Node);
-- Dynamic index for Synth_Assignment_Prefix.
-- As dynamic is about dynamic (!) index, the index is a net.
type Dyn_Name is record
-- Start and type of the indexed part, which can be a part of the
-- base name.
Pfx_Off : Value_Offsets;
Pfx_Typ : Type_Acc;
-- Variable offset.
Voff : Net;
end record;
No_Dyn_Name : constant Dyn_Name := (Pfx_Off => No_Value_Offsets,
Pfx_Typ => null,
Voff => No_Net);
-- Transform PFX into DEST_*.
-- DEST_BASE is the base object (with its own typ). Can be the result,
-- a net or an object larger than the result.
-- DEST_TYP is the type of the result.
-- DEST_OFF is the offset, within DEST_DYN.
-- DEST_DYN is set (Voff field set) when there is a non-static index.
procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc;
Pfx : Node;
Dest_Base : out Valtyp;
Dest_Typ : out Type_Acc;
Dest_Off : out Value_Offsets;
Dest_Dyn : out Dyn_Name);
-- Simplified version. No dynamic offset expected.
procedure Synth_Assignment_Prefix (Syn_Inst : Synth_Instance_Acc;
Pfx : Node;
Dest_Base : out Valtyp;
Dest_Typ : out Type_Acc;
Dest_Off : out Value_Offsets);
-- Likewise but for a formal name.
procedure Synth_Individual_Formal (Syn_Inst : Synth_Instance_Acc;
Formal : Type_Acc;
Pfx : Node;
Dest_Typ : out Type_Acc;
Dest_Off : out Value_Offsets);
procedure Synth_Assignment (Syn_Inst : Synth_Instance_Acc;
Target : Node;
Val : Valtyp;
Loc : Node);
function Synth_Read_Memory (Syn_Inst : Synth_Instance_Acc;
Obj : Valtyp;
Res_Typ : Type_Acc;
Off : Uns32;
Dyn : Dyn_Name;
Loc : Node) return Valtyp;
function Synth_User_Function_Call
(Syn_Inst : Synth_Instance_Acc; Expr : Node) return Valtyp;
-- Operation implemented by a user function.
function Synth_User_Operator (Syn_Inst : Synth_Instance_Acc;
Left_Expr : Node;
Right_Expr : Node;
Expr : Node) return Valtyp;
-- Generate netlists for concurrent statements STMTS.
procedure Synth_Concurrent_Statements
(Syn_Inst : Synth_Instance_Acc; Stmts : Node);
-- Apply attributes of UNIT.
procedure Synth_Attribute_Values
(Syn_Inst : Synth_Instance_Acc; Unit : Node);
procedure Synth_Verification_Unit (Syn_Inst : Synth_Instance_Acc;
Unit : Node;
Parent_Inst : Synth_Instance_Acc);
procedure Execute_Report_Statement (Inst : Synth_Instance_Acc;
Stmt : Node);
procedure Exec_Failed_Assertion (Syn_Inst : Synth_Instance_Acc;
Stmt : Node);
procedure Init_For_Loop_Statement (Inst : Synth_Instance_Acc;
Stmt : Node;
Val : out Valtyp);
procedure Finish_For_Loop_Statement (Inst : Synth_Instance_Acc;
Stmt : Node);
procedure Synth_Variable_Assignment (Inst : Synth_Instance_Acc;
Stmt : Node);
procedure Synth_Conditional_Variable_Assignment
(Inst : Synth_Instance_Acc; Stmt : Node);
procedure Synth_Procedure_Call (Syn_Inst : Synth_Instance_Acc; Stmt : Node);
procedure Synth_Subprogram_Back_Association
(Subprg_Inst : Synth_Instance_Acc;
Caller_Inst : Synth_Instance_Acc;
Inter_Chain : Node;
Assoc_Chain : Node);
function Synth_Association_Conversion (Inst : Synth_Instance_Acc;
Func : Node;
Val : Valtyp;
Res_Typ : Type_Acc) return Valtyp;
-- For simulation.
function Exec_Resolution_Call (Syn_Inst : Synth_Instance_Acc;
Imp : Node;
Obj : Node;
Arg : Valtyp) return Valtyp;
-- Return the associated choice from CHOICES chain selected by SEL.
-- It returns the choice (not the associated expression or chain) which
-- carries the association.
function Execute_Static_Choices_Scalar
(Inst : Synth_Instance_Acc; Choices : Node; Sel : Int64) return Node;
-- Return the statements chain to be executed.
function Execute_Static_Case_Statement
(Inst : Synth_Instance_Acc; Stmt : Node; Sel : Valtyp) return Node;
type Target_Kind is
(
-- The target is an object or a static part of it.
Target_Simple,
-- The target is an aggregate.
Target_Aggregate,
-- The assignment is dynamically indexed.
Target_Memory
);
type Target_Info (Kind : Target_Kind := Target_Simple) is record
-- In all cases, the type of the target is known or computed.
Targ_Type : Type_Acc;
case Kind is
when Target_Simple =>
-- For a simple target, the destination is known.
Obj : Valtyp;
Off : Value_Offsets;
when Target_Aggregate =>
-- For an aggregate: the type is computed and the details will
-- be handled at the assignment.
Aggr : Node;
when Target_Memory =>
-- For a memory: the destination is known.
Mem_Obj : Valtyp;
-- The dynamic offset.
Mem_Dyn : Dyn_Name;
-- Offset of the data to be accessed from the memory.
Mem_Doff : Uns32;
end case;
end record;
function Synth_Target (Syn_Inst : Synth_Instance_Acc;
Target : Node) return Target_Info;
-- Split aggregate assignment into smaller parts.
generic
with procedure Assign (Inst : Synth_Instance_Acc;
Targ_Info : Target_Info;
Val : Valtyp;
Loc : Node);
procedure Assign_Aggregate (Inst : Synth_Instance_Acc;
Target : Node;
Target_Typ : Type_Acc;
Val : Valtyp;
Loc : Node);
type Assoc_Record is record
Formal : Node;
Form_Off : Value_Offsets;
Act_Base : Valtyp;
Act_Typ : Type_Acc;
Act_Off : Value_Offsets;
Act_Dyn : Dyn_Name;
end record;
type Assoc_Array is array (Natural range <>) of Assoc_Record;
-- For simulation: create a value for individual signal associations.
type Create_Value_For_Signal_Individual_Assocs_Acc is
access function (Inst : Synth_Instance_Acc;
Assocs : Assoc_Array;
Typ : Type_Acc) return Valtyp;
Hook_Create_Value_For_Signal_Individual_Assocs :
Create_Value_For_Signal_Individual_Assocs_Acc;
private
-- There are 2 execution mode:
-- * static: it is like simulation, all the inputs are known, neither
-- gates nor signals are generated. This mode is used during
-- elaboration and when all inputs of a subprogram are known.
-- * dynamic: inputs can be wires so gates are generated. But many types
-- (like file or access) cannot be handled.
type Mode_Type is (Mode_Static, Mode_Dynamic);
type Loop_Context (Mode : Mode_Type);
type Loop_Context_Acc is access all Loop_Context;
type Loop_Context (Mode : Mode_Type) is record
Prev_Loop : Loop_Context_Acc;
Loop_Stmt : Node;
case Mode is
when Mode_Dynamic =>
-- Set when this loop has next/exit statements for itself.
-- Set to true so that inner loops have to declare W_Quit.
Need_Quit : Boolean;
-- Value of W_En at the entry of the loop.
Saved_En : Net;
-- Set to 0 in case of exit for the loop.
-- Set to 0 in case of exit/next for outer loop.
-- Initialized to 1.
W_Exit : Wire_Id;
-- Set to 0 if this loop has to be quited because of an
-- exit/next for an outer loop. Initialized to 1.
W_Quit : Wire_Id;
-- Mark to release wires.
Wire_Mark : Wire_Id;
when Mode_Static =>
S_Exit : Boolean;
S_Quit : Boolean;
end case;
end record;
-- Context for sequential statements.
type Seq_Context (Mode : Mode_Type) is record
Inst : Synth_Instance_Acc;
Cur_Loop : Loop_Context_Acc;
Ret_Value : Valtyp;
Ret_Typ : Type_Acc;
Nbr_Ret : Int32;
case Mode is
when Mode_Dynamic =>
-- Enable execution. For loop controls.
W_En : Wire_Id;
W_Ret : Wire_Id;
-- Return value.
W_Val : Wire_Id;
Ret_Init : Net;
when Mode_Static =>
S_En : Boolean;
end case;
end record;
end Synth.Vhdl_Stmts;
|