-- Environment definition for 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, write to the Free Software -- Foundation, Inc., 51 Franklin Street - Fifth Floor, Boston, -- MA 02110-1301, USA. with Types; use Types; with Dyn_Tables; with Tables; with Netlists; use Netlists; with Netlists.Utils; use Netlists.Utils; with Netlists.Builders; with Synth.Source; package Synth.Environment is -- A simple signal/variable is either a bit or a std_ulogic -- signal/variable, or a bus (bit_vector, std_ulogic_vector, signed, -- unsigned...). -- -- Complex signals/variables (records, arrays) are decomposed to simple -- signals/variables. -- -- Each simple signal/variable is represented by a Wire_Id. Synthesis -- deals only with these wires or group of them. type Wire_Id is private; No_Wire_Id : constant Wire_Id; function Is_Lt (L, R : Wire_Id) return Boolean; -- A Wire is either a signal, a variable or a port. We need to know the -- nature of a wire as the assignment semantic is not the same (a variable -- assignment overwrite the old value, while a signal assignment is -- effective at the next cycle). type Wire_Kind is ( Wire_None, Wire_Signal, Wire_Variable, Wire_Input, Wire_Output, Wire_Inout ); type Seq_Assign is private; No_Seq_Assign : constant Seq_Assign; type Conc_Assign is private; No_Conc_Assign : constant Conc_Assign; -- Create a wire. function Alloc_Wire (Kind : Wire_Kind; Obj : Source.Syn_Src) return Wire_Id; -- Mark the wire as free. procedure Free_Wire (Wid : Wire_Id); -- Simple mark & release. This is a very simple mechanism (will free -- all wires allocated after the mark), but efficient and working well -- for the stack based allocation. procedure Mark (M : out Wire_Id); procedure Release (M : in out Wire_Id); -- Check that all the wires have been released. procedure All_Released; -- Remove wires WID1 and WID2 from current phi. -- Used for internal wires (exit/quit) when exiting their scope. procedure Phi_Discard_Wires (Wid1 : Wire_Id; Wid2 : Wire_Id); -- Set the gate for a wire. -- The gate represent the current value. It is usually an Id_Signal. procedure Set_Wire_Gate (Wid : Wire_Id; Gate : Net); function Get_Wire_Gate (Wid : Wire_Id) return Net; -- The current value of WID. For variables, this is the last assigned -- value. For signals, this is the initial value. function Get_Current_Value (Ctxt : Builders.Context_Acc; Wid : Wire_Id) return Net; function Get_Current_Assign_Value (Ctxt : Builders.Context_Acc; Wid : Wire_Id; Off : Uns32; Wd : Width) return Net; -- Read and write the mark flag. function Get_Wire_Mark (Wid : Wire_Id) return Boolean; procedure Set_Wire_Mark (Wid : Wire_Id; Mark : Boolean := True); type Phi_Id is private; No_Phi_Id : constant Phi_Id; function Get_Wire_Id (W : Seq_Assign) return Wire_Id; function Get_Assign_Chain (Asgn : Seq_Assign) return Seq_Assign; function Get_Assign_Value (Ctxt : Builders.Context_Acc; Asgn : Seq_Assign) return Net; type Phi_Type is private; -- Create a new phi context. procedure Push_Phi; procedure Pop_Phi (Phi : out Phi_Type); -- Destroy the current phi context and merge it. Can apply only for the -- first non-top level phi context. procedure Pop_And_Merge_Phi (Ctxt : Builders.Context_Acc; Stmt : Source.Syn_Src); -- All assignments in PHI to wires below MARK are propagated to the -- current phi. Used to propagate assignments to wires defined out of -- a subprogram when leaving a subprogram. procedure Propagate_Phi_Until_Mark (Ctxt : Builders.Context_Acc; Phi : Phi_Type; Mark : Wire_Id); -- Handle if statement. According to SEL, the value of the wires are -- those from T or from F. procedure Merge_Phis (Ctxt : Builders.Context_Acc; Sel : Net; T, F : Phi_Type; Stmt : Source.Syn_Src); -- Sort all seq assign of P by wire id. Used to more easily merge them. function Sort_Phi (P : Phi_Type) return Seq_Assign; -- In the current phi context, assign VAL to DEST. procedure Phi_Assign (Ctxt : Builders.Context_Acc; Dest : Wire_Id; Val : Net; Offset : Uns32); -- Get current phi context. function Current_Phi return Phi_Id; pragma Inline (Current_Phi); procedure Add_Conc_Assign (Wid : Wire_Id; Val : Net; Off : Uns32; Stmt : Source.Syn_Src); procedure Finalize_Assignments (Ctxt : Builders.Context_Acc); -- For low-level phi merge. type Partial_Assign is private; No_Partial_Assign : constant Partial_Assign; function Get_Assign_Partial (Asgn : Seq_Assign) return Partial_Assign; function New_Partial_Assign (Val : Net; Offset : Uns32) return Partial_Assign; type Partial_Assign_Array is array (Int32 range <>) of Partial_Assign; type Partial_Assign_List is limited private; procedure Partial_Assign_Init (List : out Partial_Assign_List); procedure Partial_Assign_Append (List : in out Partial_Assign_List; Pasgn : Partial_Assign); procedure Merge_Partial_Assigns (Ctxt : Builders.Context_Acc; W : Wire_Id; List : in out Partial_Assign_List); -- P is an array of Partial_Assign. Each element is a list -- of partial assign from a different basic block. -- Extract the value to nets N of the maximal partial assignment starting -- at offset OFF for all partial assignments. Fully handled partial -- assignments are poped. Set the offset and width to OFF and WD of the -- result. procedure Extract_Merge_Partial_Assigns (Ctxt : Builders.Context_Acc; P : in out Partial_Assign_Array; N : out Net_Array; Off : in out Uns32; Wd : out Width); -- A const wire is a wire_signal which has one whole (same width as the -- wire) assignment and whose assignment value is a const net. -- That's rather restrictive but still efficient. function Is_Const_Wire (Wid : Wire_Id) return Boolean; -- Return the corresponding net for a constant wire. function Get_Const_Wire (Wid : Wire_Id) return Net; private type Wire_Id is new Uns32; No_Wire_Id : constant Wire_Id := 0; function Is_Lt (L, R : Wire_Id) return Boolean renames "<"; type Seq_Assign is new Uns32; No_Seq_Assign : constant Seq_Assign := 0; type Partial_Assign is new Uns32; No_Partial_Assign : constant Partial_Assign := 0; type Partial_Assign_List is record First, Last : Partial_Assign; end record; type Conc_Assign is new Uns32; No_Conc_Assign : constant Conc_Assign := 0; type Phi_Id is new Uns32; No_Phi_Id : constant Phi_Id := 0; type Wire_Id_Record is record -- Kind of wire: signal, variable... -- Set at initialization and cannot be changed. -- Used to know what is the current value of the wire (could be either -- Gate when it is a signal or Cur_Assign when it is a variable). Kind : Wire_Kind; -- Used in various algorithms: a flag on a wire. This flag must be -- cleared after usage. Mark_Flag : Boolean; -- Source node that created the wire. Decl : Source.Syn_Src; -- The initial net for the wire. -- This is a pseudo gate that is needed because the value of the wire -- can be read before anything was assigned to it. Gate : Net; -- Current assignment (if there is one). -- This is needed so that the current value (for variable) can be read. Cur_Assign : Seq_Assign; -- Chain of concurrent assigns for this wire. -- This is used to detect multiple collision and to handle partial -- assignments. Final_Assign : Conc_Assign; Nbr_Final_Assign : Natural; end record; type Seq_Assign_Record is record -- Target of the assignment. Id : Wire_Id; -- Assignment in the previous phi context. -- Used to restore Cur_Assign of the wire when the phi context is poped. Prev : Seq_Assign; -- Corresponding phi context for this wire. Phi : Phi_Id; -- Next wire in the phi context. Chain : Seq_Assign; -- Values assigned. Asgns : Partial_Assign; end record; type Partial_Assign_Record is record Next : Partial_Assign; -- Assignment at OFFSET. The width is set by the width of the value. Value : Net; Offset : Uns32; end record; type Conc_Assign_Record is record Next : Conc_Assign; -- Concurrent assignment at OFFSET. The width is set by value width. Value : Net; Offset : Uns32; -- Source of the assignment. Useful to report errors. Stmt : Source.Syn_Src; end record; type Phi_Type is record -- Chain of sequential assignments in the current phi context (BB). First : Seq_Assign; -- Number of assignments. Nbr : Uns32; end record; package Phis_Table is new Tables (Table_Component_Type => Phi_Type, Table_Index_Type => Phi_Id, Table_Low_Bound => No_Phi_Id, Table_Initial => 16); package Wire_Id_Table is new Tables (Table_Component_Type => Wire_Id_Record, Table_Index_Type => Wire_Id, Table_Low_Bound => No_Wire_Id, Table_Initial => 1024); package Assign_Table is new Tables (Table_Component_Type => Seq_Assign_Record, Table_Index_Type => Seq_Assign, Table_Low_Bound => No_Seq_Assign, Table_Initial => 1024); package Partial_Assign_Table is new Tables (Table_Component_Type => Partial_Assign_Record, Table_Index_Type => Partial_Assign, Table_Low_Bound => No_Partial_Assign, Table_Initial => 1024); package Conc_Assign_Table is new Tables (Table_Component_Type => Conc_Assign_Record, Table_Index_Type => Conc_Assign, Table_Low_Bound => No_Conc_Assign, Table_Initial => 1024); package Net_Tables is new Dyn_Tables (Table_Component_Type => Net, Table_Index_Type => Int32, Table_Low_Bound => 1, Table_Initial => 32); end Synth.Environment;