2010-07-12 10:33:22 -05:00
#!/usr/bin/python
# This file is part of the Coriolis Project.
# Copyright (C) Laboratoire LIP6 - Departement ASIM
# Universite Pierre et Marie Curie
#
# Main contributors :
# Christophe Alexandre <Christophe.Alexandre@lip6.fr>
# Sophie Belloeil <Sophie.Belloeil@lip6.fr>
# Hugo Clement <Hugo.Clement@lip6.fr>
# Jean-Paul Chaput <Jean-Paul.Chaput@lip6.fr>
# Damien Dupuis <Damien.Dupuis@lip6.fr>
# Christian Masson <Christian.Masson@lip6.fr>
# Marek Sroka <Marek.Sroka@lip6.fr>
#
# The Coriolis Project 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.
#
# The Coriolis Project 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 the Coriolis Project; if not, write to the Free Software
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
# USA
#
# License-Tag
# Authors-Tag
# ===================================================================
#
# x-----------------------------------------------------------------x
# | |
# | C O R I O L I S |
# | S t r a t u s - Netlists/Layouts Description |
# | |
# | Author : Sophie BELLOEIL |
# | E-mail : Sophie.Belloeil@asim.lip6.fr |
# | =============================================================== |
# | Py Module : "./st_model.py" |
# | *************************************************************** |
# | U p d a t e s |
# | |
# x-----------------------------------------------------------------x
import CRL
2010-11-16 08:09:53 -06:00
import Viewer
from Hurricane import *
2010-07-12 10:33:22 -05:00
import re , types , string
FRAMEWORK = None
EDITOR = None
MODELMAP = { }
CELLS = [ ]
STRATUS = 100
LOGICAL = CRL . Catalog . State . Logical
PHYSICAL = CRL . Catalog . State . Physical
VIEWS = CRL . Catalog . State . Views
def setEditor ( editor ) :
global EDITOR
EDITOR = editor
return
#######################
##### Class Model #####
#######################
class Model :
##########################
##### Initialisation #####
##########################
def __init__ ( self , nom , param = { } , hurCell = None , hurricane_plug = True ) :
global FRAMEWORK , CELLS
2010-09-06 04:12:24 -05:00
# Look up the editor
if globals ( ) . has_key ( " __editor " ) : setEditor ( __editor )
2010-07-12 10:33:22 -05:00
self . _name = nom
self . _param = param
self . _hur_cell = hurCell
self . _st_insts = [ ] # Tab : list of the instances
self . _st_ports = [ ] # Tab : list of the nets
self . _st_sigs = [ ] # Tab : list of the nets
self . _st_partsigs = [ ] # Tab : list of the partial nets
self . _st_merge = [ ] # Tab : list of the nets to merge, in order to have them in the right order
self . _hierarchy = 0 # Modified in ModelCreation
self . _TAB_NETS_OUT = [ ]
self . _TAB_NETS_CAT = [ ]
self . _NB_INST = 0
# List of the generators used for overload
self . _and = ' A2 '
self . _or = ' O2 '
self . _xor = ' Xr2 '
self . _not = ' Inv '
self . _buff = ' Buf '
self . _reg = ' Sff1 '
# Stratus generators
self . _mux = ' Smux '
self . _shift = ' Shifter '
self . _comp = ' Comp '
# Arithmetic generators
self . _add = ' Slansky '
self . _sub = ' Sub '
self . _mult = ' Multiplier '
self . _div = None
self . _signed = True
self . _extended = False
self . _model_map = { }
# List of the under-cells of a cell
self . _underCells = { }
# self._tabPlacement = []
# Instance reference for placement
self . _insref = None
# Place and Route
self . _nb_alims_verticales = 0
self . _nb_pins = 0 # to number the pins of the core
self . _nb_vdd_pins = 0 # to number the pins vdd of the core
self . _nb_vss_pins = 0 # to number the pins vss of the core
self . standard_instances_list = [ ]
self . pad_north = [ ] # list to save the pads names
self . pad_south = [ ]
self . pad_east = [ ]
self . pad_west = [ ]
# Creation of the database
if not FRAMEWORK :
#Initialize()
2010-07-17 05:34:21 -05:00
FRAMEWORK = CRL . AllianceFramework . get ( )
2010-07-12 10:33:22 -05:00
self . _st_vdds = [ ]
self . _st_vsss = [ ]
if hurCell :
from st_net import VddInFromHur
from st_net import VssInFromHur
2010-09-06 04:12:24 -05:00
try :
netVdd = iter ( hurCell . getPowerNets ( ) ) . next ( )
2010-09-06 06:22:30 -05:00
self . _st_vdds . append ( VddInFromHur ( netVdd ) )
2010-09-06 04:12:24 -05:00
except StopIteration :
print " [Stratus Warning] : Cell " , self . _name , " does not have a vdd port. "
pass
try :
netVss = iter ( hurCell . getGroundNets ( ) ) . next ( )
2010-09-06 06:22:30 -05:00
self . _st_vsss . append ( VssInFromHur ( netVss ) )
2010-09-06 04:12:24 -05:00
except StopIteration :
print " [Stratus Warning] : Cell " , self . _name , " does not have a vss port. "
pass
2010-07-12 10:33:22 -05:00
self . _st_cks = [ ]
if not hurCell :
# List of the cells
CELLS . append ( self )
# List of the undercells
if CELLS [ 0 ] != self :
CELLS [ 0 ] . _underCells [ self ] = 0
# CELLS[0]._tabPlacement.append ( self )
self . _hur_plug = hurricane_plug
# HurricanePLug if : no hurCell, in mode hurricane plug (i.e. hurricane_plug and hurricane_plug of the masterCell)
if ( not hurCell ) and ( hurricane_plug ) :
if CELLS [ 0 ] == self :
self . HurricanePlug ( )
else :
if CELLS [ 0 ] . _hur_plug :
self . HurricanePlug ( )
##### Delete #####
def Delete ( self ) :
UpdateSession . open ( )
under_cells = self . _underCells
self . _hur_cell . Delete ( )
for cell in under_cells : cell . _hur_cell . Delete ( )
UpdateSession . close ( )
#####################################################
##### Initialisation of attributes for the cell #####
#####################################################
def InitGraph ( self ) :
self . _graph = True
# External nets
for net in self . _st_ports :
net . _st_inst_in = None
net . _st_insts_out = [ ]
net . _covered = False
# Internal nets
for net in self . _st_sigs :
net . _st_inst_in = None
net . _st_insts_out = [ ]
net . _covered = False
# Part nets
for pnet in self . _st_partsigs :
pnet . _st_inst_in = None
pnet . _st_insts_out = [ ]
pnet . _covered = False
# Instances
for inst in self . _st_insts :
inst . _st_nets_in = [ ]
inst . _st_nets_out = [ ]
# Initialisation of instances and nets
self . initNet ( inst )
self . initInst ( inst )
########################################################
##### Initialisation of the attributes of the nets #####
########################################################
def initNet ( self , instance ) :
# For each pin
for pin in instance . _map :
# Check if the pin is an output or an input
direct = " "
direct = self . getDirect ( instance . _model , pin )
# Get the net connected to this pin
if instance . _map [ pin ] . _to_merge : mapNet = instance . _map [ pin ] . _to_merge [ 0 ] [ 0 ]
else : mapNet = instance . _map [ pin ]
if len ( instance . _map [ pin ] . _to_merge ) > mapNet . _arity :
mapNet . _real_arity = len ( instance . _map [ pin ] . _to_merge )
mapNet . _inverse_merge = instance . _map [ pin ]
# Initialisation
if direct == ' input ' :
if mapNet . _real_net :
mapNet = mapNet . _real_net
if ' _st_insts_out ' not in mapNet . __dict__ : mapNet . _st_insts_out = [ ]
if instance not in mapNet . _st_insts_out : mapNet . _st_insts_out + = [ instance ]
elif direct == ' output ' : # FIXME _real_net not taken care of
mapNet . _st_inst_in = instance
#############################################################
##### Initialisation of the attributes of the instances #####
#############################################################
def initInst ( self , instance ) :
# For each pin
for pin in instance . _map :
# Check if the pin is an output or an input
direct = " "
direct = self . getDirect ( instance . _model , pin )
# Get the net connected to this pin
if instance . _map [ pin ] . _to_merge : mapNet = instance . _map [ pin ] . _to_merge [ 0 ] [ 0 ]
else : mapNet = instance . _map [ pin ]
# Initialisation
if direct == ' input ' : instance . _st_nets_in + = [ mapNet ]
elif direct == ' output ' : instance . _st_nets_out + = [ mapNet ]
###########################################
##### Returns the direction of a port #####
###########################################
def getDirect ( self , model , pin ) :
global FRAMEWORK
hurCell = FRAMEWORK . getCell ( model , CRL . Catalog . State . Views )
stCell = None
if not hurCell :
for c in self . _underCells :
if c . _name == model :
stCell = c
break
if not hurCell and not stCell :
2010-07-22 09:33:55 -05:00
err = " \n Error : no cell found with model %s . \n " % model
raise Exception ( err )
2010-07-12 10:33:22 -05:00
# Hurricane cell found:
if hurCell :
found = False
2011-02-16 07:59:03 -06:00
for net in hurCell . getExternalNets ( ) :
2010-07-12 10:33:22 -05:00
direction = net . getDirection ( )
name = str ( net . getName ( ) )
if net . getType ( ) in ( TypePOWER , TypeGROUND , TypeCLOCK ) :
found = True
continue
chaine = re . search ( " (.*) \ ([0-9]+ \ ) " , name )
if chaine : name = chaine . group ( 1 )
if name == pin :
found = True
if direction == DirectionOUT : return " output "
elif direction == DirectionIN : return " input "
2010-07-22 09:33:55 -05:00
else :
err = " \n Error : unable to find direction of port %s in model %s . Direction is %d \n " % ( pin , model , direction )
raise Exception ( err )
2010-07-12 10:33:22 -05:00
2010-07-22 09:33:55 -05:00
if not found :
err = " \n Error : unable to find port %s in model %s . \n " % ( pin , model )
raise Exception ( err )
2010-07-12 10:33:22 -05:00
# Stratus cell found:
elif stCell :
found = False
for net in stCell . _st_vdds :
if net . _name == pin :
found = True
break
for net in stCell . _st_vsss :
if net . _name == pin :
found = True
break
for net in stCell . _st_cks :
if net . _name == pin :
found = True
break
# _st_ports i.e. not alimentations and not clock
for net in stCell . _st_ports :
if net . _name == pin :
found = True
if net . _direct == " OUT " : return " output "
elif net . _direct == " IN " : return " input "
2010-07-22 09:33:55 -05:00
else :
err = " \n Error : unable to find direction of port %s in model %s . \n " % ( pin , model )
raise Exception ( err )
2010-07-12 10:33:22 -05:00
if not found :
ports = " "
for net in stCell . _st_ports : ports + = net . _name + " , "
2010-07-22 09:33:55 -05:00
err = " \n Error : unable to find port %s in model %s . \n Ports are : %s \n " % ( pin , model , ports )
raise Exception ( err )
2010-07-12 10:33:22 -05:00
else :
err = " \n [ERROR] InitGraph : no model named " + model + " in the database. \n "
2010-07-22 09:33:55 -05:00
raise Exception ( err )
2010-07-12 10:33:22 -05:00
#############################
##### Print of the cell #####
#############################
def Print ( self ) :
print " ################## The Cell ################## "
for inst in self . _st_insts :
print " * inst name : " , inst . _name
print " inst model : " , inst . _model
for pin in inst . _map :
if pin != ' vdd ' and pin != ' vss ' :
net = inst . _map [ pin ]
print " pin : " , pin , " net : " , net . _name , " with arity : " , net . _arity
if net . _to_merge :
net = inst . _map [ pin ] . _to_merge [ 0 ] [ 0 ]
print " net merged with : " , net . _name , " with arity : " , net . _arity
2010-11-16 08:09:53 -06:00
#############################
##### Find An Instance #####
#############################
def getInstance ( self , name ) :
for instance in self . _st_insts :
if instance . _name == name :
return instance
return None
2010-07-12 10:33:22 -05:00
#########################
def PrintGraph ( self ) :
if " _graph " not in self . __dict__ :
err = " \n [Stratus ERROR] PrintGraph : The graph does not exist. Use initGraph before. \n "
2010-07-22 09:33:55 -05:00
raise Exception ( err )
2010-07-12 10:33:22 -05:00
else :
if self . _graph == False :
err = " \n [Stratus ERROR] PrintGraph : The graph does not exist. Use initGraph before. \n "
2010-07-22 09:33:55 -05:00
raise Exception ( err )
2010-07-12 10:33:22 -05:00
print " ################## Cell ' s Graph features ################## "
for inst in self . _st_insts :
print " => inst name : " , inst . _name , " ( model : " , inst . _model , " ) "
print " inst st_nets_in : "
for net in inst . _st_nets_in : print " " , net . _name
print " inst st_nets_out : "
for net in inst . _st_nets_out : print " " , net . _name
for port in self . _st_ports :
print " => port name : " , port . _name
if port . _st_inst_in : print " port _st_inst_in : " , port . _st_inst_in . _name
print " port _st_insts_out : "
for p in port . _st_insts_out : print " " , p . _name
for sig in self . _st_sigs :
print " => sig name : " , sig . _name
if sig . _st_inst_in : print " sig _st_inst_in : " , sig . _st_inst_in . _name
print " sig _st_insts_out : "
for s in sig . _st_insts_out : print " " , s . _name
for psig in self . _st_partsigs :
print " => part sig name : " , psig . _name
if ' _st_inst_in ' in psig . __dict__ and psig . _st_inst_in : print " part sig _st_inst_in : " , psig . _st_inst_in . _name
if ' _st_insts_out ' in psig . __dict__ :
print " part sig _st_insts_out : "
for s in psig . _st_insts_out : print " " , s . _name
#####################
##### overloard #####
#####################
##### __getattr #####
# def __getattr__ ( self, attr ) :
##### __str #####
def __str__ ( self ) :
s = " Model " + str ( self . _hur_cell )
return s
########################
##### VLSI Methods #####
########################
##### Interface #####
def Interface ( self ) :
print " [Stratus Warning] : Execution of empty Interface method for " , self . _name , " . "
##### Netlist #####
def Netlist ( self ) :
print " [Stratus Warning] : Execution of empty Netlist method for " , self . _name , " . "
pass
##### Layout #####
def Layout ( self ) :
print " [Stratus Warning] : Execution of empty Layout method for " , self . _name , " . "
pass
##### Vbe #####
def Vbe ( self ) :
self . _vbe = True
##### View : in order to see the in the editor #####
2011-02-16 07:59:03 -06:00
def View ( self , stopLevel = 1 , message = " Status stop point " ) :
2010-07-12 10:33:22 -05:00
global EDITOR
if not self . _hur_cell :
err = " \n [Stratus ERROR] View : Hurricane Cell does not exist. \n Check CRL_IN_LO/CRL_IN_PH variables. \n "
2010-07-22 09:33:55 -05:00
raise Exception ( err )
2010-07-12 10:33:22 -05:00
if EDITOR :
EDITOR . setCell ( self . _hur_cell )
EDITOR . fit ( )
Breakpoint . stop ( stopLevel , message )
##### Save : in order to create the output files #####
def Save ( self , views = 0 , fileName = None ) :
global FRAMEWORK
global CELLS
if views == STRATUS :
self . exportStratus ( fileName )
else :
UpdateSession . open ( )
hurCell = self . _hur_cell
if str ( hurCell . getName ( ) ) != " __Scratch__ " :
FRAMEWORK . saveCell ( hurCell , views | CRL . Catalog . State . Logical )
if len ( CELLS ) == 0 :
err = " \n [Stratus ERROR] Save : CELLS stack is empty. \n "
2010-07-22 09:33:55 -05:00
raise Exception ( err )
2010-07-12 10:33:22 -05:00
CELLS . pop ( )
UpdateSession . close ( )
##### Simul : in order to use simulation tool #####
def Simul ( self , name = None , tool = ' asimut ' ) :
from utils import runpat
if not name : name = self . _name
2011-02-16 07:59:03 -06:00
if tool == ' asimut ' : runpat ( self . _name , name , ' -l 1 -p 100 -zerodelay -nocheckdriver -nostrict -bdd -nowarning ' )
2010-07-22 09:33:55 -05:00
else : raise Exception ( ' not implemented yet ' )
2010-07-12 10:33:22 -05:00
##### Create a stratus file given the database #####
def exportStratus ( self , fileName ) :
## Cat formating ##
def catName ( net , tableau ) :
name = " Cat ( "
paire = tableau [ - 1 ]
if paire :
net1 = paire [ 0 ]
bit1 = str ( paire [ 1 ] )
else :
net1 = net
bit1 = str ( len ( tableau ) - 1 )
for i in range ( len ( tableau ) - 2 , - 1 , - 1 ) :
change = False
if tableau [ i ] and not tableau [ i + 1 ] :
change = True
elif not tableau [ i ] and tableau [ i + 1 ] :
change = True
elif tableau [ i ] and tableau [ i + 1 ] :
if tableau [ i ] [ 0 ] != tableau [ i + 1 ] [ 0 ] :
change = True
if change :
if tableau [ i + 1 ] : bit2 = str ( tableau [ i + 1 ] [ 1 ] )
else : bit2 = str ( i + 1 )
name + = " self. "
if int ( bit2 ) == int ( net1 . _ind ) and int ( bit1 ) == ( net1 . _arity - 1 ) :
name + = net1 . _name
else :
name + = net1 . _name
name + = " [ "
name + = bit1
name + = " : "
name + = bit2
name + = " ] "
name + = " , "
if tableau [ i ] : net1 = tableau [ i ] [ 0 ]
else : net1 = net
if tableau [ i ] : bit1 = str ( tableau [ i ] [ 1 ] )
else : bit1 = str ( i )
if tableau [ 0 ] :
bit2 = str ( tableau [ 0 ] [ 1 ] )
else :
bit2 = " 0 "
name + = " self. "
if int ( bit2 ) == int ( net1 . _ind ) and int ( bit1 ) == ( net1 . _arity - 1 ) :
name + = net1 . _name
else :
name + = net1 . _name
name + = " [ "
name + = bit1
name + = " : "
name + = bit2
name + = " ] "
name + = " ) "
return name
#########################
nom = str ( self . __class__ )
chaine = re . search ( " [^ \ .]* \ .(.*) " , nom )
nom = chaine . group ( 1 )
if not fileName : fileName = nom + " _created "
fileName + = " .py "
file = open ( fileName , " w+ " )
file . write ( " #!/usr/bin/python \n \n " )
file . write ( " from stratus import * \n \n " )
file . write ( " class %s ( Model ) : \n \n " % nom )
##### Interface #####
file . write ( " def Interface ( self ) : \n \n " )
for net in self . _st_ports :
if net . _direct == " IN " : classe = " SignalIn "
elif net . _direct == " OUT " : classe = " SignalOut "
elif net . _direct == " INOUT " : classe = " SignalInOut "
file . write ( " self. %s = %s ( \" %s \" , %d ) \n " % ( net . _name , classe , net . _name , net . _arity ) )
file . write ( " \n " )
for ck in self . _st_cks :
if ck . _ext :
file . write ( " self. %s = CkIn ( \" %s \" ) \n " % ( ck . _name , ck . _name ) )
file . write ( " self.vdd = VddIn ( \" vdd \" ) \n " )
file . write ( " self.vss = VssIn ( \" vss \" ) \n " )
file . write ( " \n " )
##### Netlist #####
file . write ( " def Netlist ( self ) : \n \n " )
# Signals
for net in self . _st_sigs :
if net . _to_merge :
towrite = False
for paire in net . _to_merge :
if paire == 0 : towrite = True
if not towrite : continue
file . write ( " self. %s = Signal ( \" %s \" , %d ) \n " % ( net . _name , net . _name , net . _arity ) )
file . write ( " \n " )
# Generate
for cell in CELLS [ 0 ] . _underCells :
if CELLS [ 0 ] . _underCells [ cell ] == 1 :
classe = str ( cell . __class__ )
nom = cell . _name
param = cell . _param
# Gestion of libraries with specific treatment
chaine = re . search ( " (.*) \ .([^ \ .]*) " , classe )
if chaine :
fichier = chaine . group ( 1 )
if re . search ( " dpgen_ " , fichier ) or re . search ( " st_ " , fichier ) :
classe = chaine . group ( 2 )
# Special treatment for virtual library
if classe == " Bool " :
nom = cell . _name
classe = cell . _param [ ' model ' ]
param = cell . _param
del param [ ' model ' ]
del param [ ' realModel ' ]
classe = string . upper ( classe [ 0 ] ) + classe [ 1 : ] #FIXME !!!
if re . search ( " (.*) \ .([^ \ .]*) " , classe ) : # ArithLib generator
classe = string . lower ( classe [ 0 ] ) + classe [ 1 : ]
file . write ( " Generate ( \" %s \" , \" %s \" , param = %s ) \n " % ( classe , nom , param ) )
file . write ( " \n " )
# Instances
for inst in self . _st_insts :
#chaine = re.search ( "([^_]*)_(.*)", inst._name ) #XTOF FIXME !!!
#instName = chaine.group(1) #XTOF FIXME !!
file . write ( " self. %s = Inst ( \" %s \" \n " % ( inst . _name , inst . _model ) )
file . write ( " , \" %s \" \n " % inst . _name )
file . write ( " , map = { " )
# Map
for pin in inst . _map :
toto = False
tata = False
netInMap = inst . _map [ pin ]
nom = " self. " + netInMap . _name
# <=
if netInMap . _to_merge :
oneperone = True
for i in range ( len ( netInMap . _to_merge ) ) :
paire = netInMap . _to_merge [ i ]
if paire :
net = paire [ 0 ]
if i > 0 :
if netInMap . _to_merge [ i - 1 ] != 0 :
netbefore = netInMap . _to_merge [ i - 1 ] [ 0 ]
else :
oneperone = False
break
if net != netbefore :
oneperone = False
break
if paire [ 1 ] != i :
oneperone = False
break
else :
toto = True
oneperone = False
break
# "easy" case : 1 per 1 corresponding
if oneperone :
netInMap = netInMap . _to_merge [ 0 ] [ 0 ]
nom = " self. " + netInMap . _name
# else : Cat of the different signals
else :
nom = catName ( netInMap , netInMap . _to_merge )
# Cat
if netInMap . _to_cat :
tata = True
nom = catName ( netInMap , netInMap . _to_cat )
if toto and tata : print " Attention est ce un cas bien gere ??? "
if pin == inst . _map . keys ( ) [ 0 ] : file . write ( " \" %s \" : %s \n " % ( pin , nom ) )
else : file . write ( " , \" %s \" : %s \n " % ( pin , nom ) )
file . write ( " } \n " )
file . write ( " ) \n \n " )
##### End #####
file . close ( )
#### Quit : to quit the current cell without saving #####
def Quit ( self ) :
global CELLS
CELLS . pop ( )
#########################################
##### GetModelName : Class's method #####
#########################################
def GetModelName ( param ) :
return None
GetModelName = staticmethod ( GetModelName )
#########################################
##### GetParam : Class's method #####
#########################################
def GetParam ( ) :
return None
GetParam = staticmethod ( GetParam )
###############
##### Set #####
###############
def SetAnd ( self , model ) : self . _and = model
def SetOr ( self , model ) : self . _or = model
def SetXor ( self , model ) : self . _xor = model
def SetNot ( self , model ) : self . _not = model
def SetBuff ( self , model ) : self . _buff = model
def SetReg ( self , model ) : self . _reg = model
def SetMux ( self , model ) : self . _mux = model
def SetShift ( self , model ) : self . _shift = model
def SetComp ( self , model ) : self . _comp = model
def SetAdd ( self , model ) : self . _add = model
def SetMult ( self , model ) : self . _mult = model
def SetDiv ( self , model ) : self . _div = model
def SetSigned ( self , model ) : self . _signed = model
def SetExtended ( self , model ) : self . _extended = model
#############################
##### Creation of model #####
#############################
def ModelCreation ( self , modele , modele2 , dict , hierarchy , realModele = None , inOut = None ) :
global CELLS
from st_parser import BVg
## Virtual library ##
from st_bool import Bool
## Stratus generators ##
from st_mux import Smux
from st_shifter import Shifter
from st_comp import Comp
from st_const import Constant
from st_extend import Extend
from st_slansky import Slansky
from st_slansky import Sub
from st_mult import Multiplier
## Genlib ##
from dpgen_Shifter import DpgenShifter
from dpgen_Shrot import DpgenShrot
from dpgen_Nul import DpgenNul
from dpgen_RAM import DpgenRam
from dpgen_ROM2 import DpgenRom2
from dpgen_ROM4 import DpgenRom4
from dpgen_ADSB2F import DpgenAdsb2f
from dpgen_RF1 import DpgenRf1 , DpgenRf1d , DpgenRf1r0 , DpgenRf1dr0 , DpgenFifo
from dpgen_RF2 import DpgenRf2 , DpgenRf2d , DpgenRf2r0 , DpgenRf2dr0
from dpgen_Regular import DpgenAnd2 , DpgenAnd3 , DpgenAnd4 , DpgenBuff , DpgenBuse , DpgenConst , DpgenDff , DpgenDfft , DpgenInv , DpgenMux2 , DpgenNand2 , DpgenNand2mask , DpgenNand3 , DpgenNand4 , DpgenNbuse , DpgenNmux2 , DpgenNor2 , DpgenNor2mask , DpgenNor3 , DpgenNor4 , DpgenOr2 , DpgenOr3 , DpgenOr4 , DpgenSff , DpgenSfft , DpgenXnor2 , DpgenXnor2mask , DpgenXor2
from util_Gen import F_MSB_FIRST
if type ( dict ) != types . DictType :
2010-07-22 09:33:55 -05:00
err = " \n [Stratus ERROR] Inst : instanciation of a user ' s defined generator. The methods ' arguments must be dictionnaries. \n "
raise Exception ( err )
2010-07-12 10:33:22 -05:00
##### Creation of the instance #####
# dict['flags'] = F_MSB_FIRST # When vst driver permits to do F_LSB_FIRST or F_LSB_FIRST TODO
## Virtual library ##
if modele in BVg :
dict [ ' model ' ] = modele . lower ( )
dict [ ' realModel ' ] = realModele
instCell = Bool ( modele2 , dict , inOut )
## Generator ##
elif re . search ( " \ . " , modele ) :
chaine = re . search ( " (.*) \ .([^ \ .]*)$ " , modele )
ficName = chaine . group ( 1 )
className = chaine . group ( 2 )
# Roselyne : pour prendre en compte les modeles recursifs
import sys
#moduleClass = __import__ ( ficName, globals(), locals(), [className] )
#modeleClass = getattr ( moduleClass, className )
try :
modeleClass = getattr ( sys . modules [ ' __main__ ' ] , className )
except AttributeError :
moduleClass = __import__ ( ficName , globals ( ) , locals ( ) , [ className ] )
modeleClass = getattr ( moduleClass , className )
# fin Roselyne
instCell = modeleClass ( modele2 , dict )
else :
instCell = eval ( " %s ( \" %s \" , %s ) " % ( modele , modele2 , str ( dict ) ) )
## MAJ of the hierarchy ##
instCell . _hierarchy = hierarchy
##### Methods #####
if ( self . _hur_cell ) and ( not instCell . _hur_cell ) : instCell . HurricanePlug ( )
instCell . Interface ( )
instCell . Netlist ( )
if ( ' clean ' in dict ) and ( dict [ ' clean ' ] != False ) :
if ' interactive ' not in dict : dict [ ' interactive ' ] = False
instCell . Clean ( dict [ ' interactive ' ] , dict [ ' clean ' ] )
if ( ' behavioral ' in dict ) and ( dict [ ' behavioral ' ] == True ) :
instCell . Vbe ( )
if instCell . _hur_cell :
v = 0
if ( ' physical ' in dict ) and ( dict [ ' physical ' ] == True ) :
instCell . Layout ( )
v = CRL . Catalog . State . Physical
instCell . Save ( v | CRL . Catalog . State . Logical )
else :
instCell . Quit ( )
if CELLS [ 0 ] != instCell : CELLS [ 0 ] . _underCells [ instCell ] = hierarchy
return instCell
#################################################################
##### HurricanePlug : in order to create the hurricane cell #####
#################################################################
def HurricanePlug ( self ) :
global FRAMEWORK , CELLS
if self . _hur_cell :
print " [Stratus Warning] : The stratus cell already exists. "
return
self . _hur_plug = True
##### Creation of all the under cells #####
# Initialisation of the max depth
max_depth = 0
for keyCell in self . _underCells :
if self . _underCells [ keyCell ] > max_depth : max_depth = self . _underCells [ keyCell ]
# Creation from the leaf to the top cell
while max_depth :
for keyCell in self . _underCells :
if self . _underCells [ keyCell ] == max_depth :
if not FRAMEWORK . getCell ( keyCell . _name , CRL . Catalog . State . Views ) :
CELLS . append ( keyCell )
keyCell . CellHurCreation ( )
max_depth - = 1
# Creation of the top cell
CELLS . append ( self )
self . CellHurCreation ( )
##### CellHurCreation #####
def CellHurCreation ( self ) :
global FRAMEWORK , CELLS
# The cell
hurCell = FRAMEWORK . createCell ( self . _name )
hurCell . setTerminal ( 0 )
self . _hur_cell = hurCell
MODELMAP [ str ( self . _hur_cell ) ] = self
# The nets
for net in self . _st_ports : net . create_hur_net ( )
for net in self . _st_sigs : net . create_hur_net ( )
for net in self . _st_cks : net . create_hur_net ( )
for net in self . _st_vdds : net . create_hur_net ( )
for net in self . _st_vsss : net . create_hur_net ( )
# The instances and the connection
for inst in self . _st_insts : inst . create_hur_inst ( inst . _model )
# Merge
for net in self . _st_merge : net . hur_merge ( )
if ( self != CELLS [ 0 ] ) and ( self . _st_insts ) : # Layout done if HurricanePlug after Netlist method
#self.Layout() # FIXME trouver une raison de le faire ou pas : instances placees ou pas ... HurricanePlug ne doit plus marcher avec placement A VERIFIER 14/01/08
if self . _st_insts [ 0 ] . _hur_instance . getPlacementStatus ( ) : self . Save ( CRL . Catalog . State . Physical )
else : self . Save ( CRL . Catalog . State . Logical )
else :
self . Quit ( )
####################
##### Clean Up #####
####################
def Clean ( self , interactive = False , tab_name_net = [ ] ) :
# Hurricane cell
hurricane_cell = self . _hur_cell
# Useful for print
TAB_NETS = [ ]
TAB_INSTS = [ ]
cpt_inst_del = 0
cpt_net_del = 0
##### Tabs initialisation #####
net_entree = [ ]
net_sortie = [ ]
net_interne = [ ]
for net in hurricane_cell . getNets ( ) :
if net . getDirection ( ) == DirectionUNDEFINED : net_interne . append ( net )
elif net . getDirection ( ) == DirectionIN : net_entree . append ( net )
elif net . getDirection ( ) == DirectionOUT : net_sortie . append ( net )
##### File initialisation #####
file = [ ]
# Output nets put in the file if there are given as arguments
if tab_name_net != True :
for net_name in tab_name_net :
for net in net_sortie :
if net_name == str ( net . getName ( ) ) :
file . insert ( 0 , net )
if interactive : print " Output Net " , net , " has to be erased, it is put in the fifo. "
## Internal nets ##
# Number of plugs of each net :
# if equal to 0 : del net
# if equal to 1 : put net in the file
for net in net_interne :
nb_plugs = self . count_plugs ( net )
if nb_plugs == 0 :
if interactive : print " * One net suppressed (a) : " , net
TAB_NETS . append ( net . getName ( ) )
cpt_net_del + = 1
net . Delete ( )
elif nb_plugs == 1 :
if net . getPlugs ( ) . next ( ) . getMasterNet ( ) . getDirection ( ) == DirectionOUT : # output of an instance
if interactive : print " * One net put in the fifo : " , net
file . insert ( 0 , net )
## Ouput nets ##
# Number of plugs of each net :
# if equal to 0 : del net
for net in net_sortie + net_entree :
cpt_plugs_sor = self . count_plugs ( net )
if cpt_plugs_sor == 0 :
print " [Stratus Warning] Clean : Interface of " , self . _name , " changed, net : " , net , " is suppressed "
TAB_NETS . append ( net . getName ( ) )
cpt_net_del + = 1
net . Delete ( )
##### Algorithm #####
while len ( file ) > 0 :
net_file = file . pop ( )
plug = net_file . getPlugs ( ) . next ( )
inst = plug . getInstance ( )
# input nets of the instance
plugs_inst = inst . getPlugs ( )
net_entree_inst = [ ]
net_sortie_inst = [ ]
cpt_sortie = 0
for plug_de_inst in plugs_inst :
# compute the number of outputs of the instance
if plug_de_inst . getMasterNet ( ) . getDirection ( ) == DirectionOUT :
cpt_sortie + = 1
# output nets of the instance put in tab, except for the net being worked on
if net_file != plug_de_inst . getNet ( ) : net_sortie_inst . append ( plug_de_inst . getNet ( ) )
if plug_de_inst . getMasterNet ( ) . getDirection ( ) == DirectionIN :
# ignore vdd and vss
type = plug_de_inst . getNet ( ) . getType ( )
if type not in ( TypePOWER , TypeGROUND ) : net_entree_inst . append ( plug_de_inst . getNet ( ) )
### Deletion of te instance ###
# If the instance has only one output
if cpt_sortie == 1 :
if interactive : print " * One net suppressed (b) : " , net_file
TAB_NETS . append ( net_file . getName ( ) )
cpt_net_del + = 1
net_file . Delete ( )
if interactive : print " * One instance suppressed (a) : " , inst , inst
TAB_INSTS . append ( inst . getName ( ) )
cpt_inst_del + = 1
UpdateSession . open ( )
inst . Delete ( )
UpdateSession . close ( )
for net_ent in net_entree_inst :
cpt_plugs_in = self . count_plugs ( net_ent )
if cpt_plugs_in == 0 :
if net_ent in net_entree :
print " [Stratus Warning] Clean : Interface of " , self . _name , " changed, net : " , net_ent , " is suppressed "
TAB_NETS . append ( net_ent . getName ( ) )
cpt_net_del + = 1
net_ent . Delete ( )
else :
if interactive : print " * One net suppressed (c) : " , net_ent
TAB_NETS . append ( net_ent . getName ( ) )
cpt_net_del + = 1
net_ent . Delete ( )
elif cpt_plugs_in == 1 :
if net_ent . getPlugs ( ) . next ( ) . getMasterNet ( ) . getDirection ( ) == DirectionOUT : # is an output net of another instance
if interactive : print " * One net put in the fifo : " , net_ent
file . insert ( 0 , net_ent )
# If the instance has more than one output
elif cpt_sortie > = 2 :
connect = False
for net_sor in net_sortie_inst :
cpt_sor = self . count_plugs ( net_sor )
if cpt_sor > = 2 :
connect = True # at least one output connected
break
elif cpt_sor == 1 :
if net_sor in net_sortie :
connect = True # at least one output connected
break
if not ( connect ) :
if interactive : print " * One net suppressed (d) : " , net_file
TAB_NETS . append ( net_file . getName ( ) )
cpt_net_del + = 1
net_file . Delete ( )
for net_sor in net_sortie_inst :
if net_sor in file : file . remove ( net_sor )
if interactive : print " * One net suppressed (e) : " , net_sor
TAB_NETS . append ( net_sor . getName ( ) )
cpt_net_del + = 1
net_sor . Delete ( )
if interactive : print " * One instance suppressed (b) : " , inst
TAB_INSTS . append ( inst . getName ( ) )
cpt_inst_del + = 1
UpdateSession . open ( )
inst . Delete ( )
UpdateSession . close ( )
for net_ent in net_entree_inst :
cpt_plugs_in = self . count_plugs ( net_ent )
if cpt_plugs_in == 0 :
if net_ent in net_entree :
print " [Stratus Warning] Clean : Interface of " , self . _name , " changed, net : " , net_ent , " is suppressed "
TAB_NETS . append ( net_ent . getName ( ) )
cpt_net_del + = 1
net_ent . Delete ( )
else :
if interactive : print " * One net suppressed (f) : " , net_ent
TAB_NETS . append ( net_ent . getName ( ) )
cpt_net_del + = 1
net_ent . Delete ( )
elif cpt_plugs_in == 1 :
if net_ent . getPlugs ( ) . next ( ) . getMasterNet ( ) . getDirection ( ) == DirectionOUT : # in an output net of another instance
if interactive : print " * One net net put in the fifo : " , net_ent
file . insert ( 0 , net_ent )
else :
if interactive : print " The net " , net_file , " can not be delayed, it may be delayed later "
else :
print " [Warning] Pb in Clean. "
if interactive :
print " "
print " * Number of net suppressed : " , cpt_net_del
print " * List of these nets : " , TAB_NETS
print " "
print " * Number of instances suppressed : " , cpt_inst_del
print " * List of these instance : " , TAB_INSTS
print " "
###############################
def count_plugs ( self , net ) :
return len ( net . getPlugs ( ) )
#############################################################################
############################## Place and route ##############################
#############################################################################
def getCore ( self ) :
''' This function returns the instance " core " : the only one which is not a pad '''
from placeandroute import isPad
cores = [ ]
for instance in self . _hur_cell . getInstances ( ) :
if not isPad ( instance ) :
cores . append ( instance )
2010-07-22 09:33:55 -05:00
if len ( cores ) == 0 : err = " \n [Stratus ERROR] getCore : No core found. \n "
elif len ( cores ) > 1 : err = " \n [Stratus ERROR] getCore : More than one core found. \n "
if len ( cores ) != 1 : raise Exception ( err )
2010-07-12 10:33:22 -05:00
return cores [ 0 ]