256 lines
8.6 KiB
Python
256 lines
8.6 KiB
Python
#!/usr/bin/env python
|
|
#
|
|
# This file is part of the Coriolis Software.
|
|
# Copyright (c) UPMC 2014-2018, All Rights Reserved
|
|
#
|
|
# +-----------------------------------------------------------------+
|
|
# | C O R I O L I S |
|
|
# | C u m u l u s - P y t h o n T o o l s |
|
|
# | |
|
|
# | Author : Jean-Paul CHAPUT |
|
|
# | E-mail : Jean-Paul.Chaput@asim.lip6.fr |
|
|
# | =============================================================== |
|
|
# | Python : "./plugins/chip/Chip.py" |
|
|
# +-----------------------------------------------------------------+
|
|
|
|
|
|
try:
|
|
import sys
|
|
import traceback
|
|
import os.path
|
|
import optparse
|
|
import math
|
|
import cProfile
|
|
import pstats
|
|
import Cfg
|
|
import Hurricane
|
|
from Hurricane import DataBase
|
|
from Hurricane import DbU
|
|
from Hurricane import Point
|
|
from Hurricane import Transformation
|
|
from Hurricane import Box
|
|
from Hurricane import Path
|
|
from Hurricane import Occurrence
|
|
from Hurricane import UpdateSession
|
|
from Hurricane import Breakpoint
|
|
from Hurricane import Net
|
|
from Hurricane import RoutingPad
|
|
from Hurricane import Contact
|
|
from Hurricane import Horizontal
|
|
from Hurricane import Vertical
|
|
from Hurricane import Instance
|
|
from Hurricane import HyperNet
|
|
from Hurricane import Query
|
|
import Viewer
|
|
import CRL
|
|
from CRL import RoutingLayerGauge
|
|
import helpers
|
|
from helpers.io import ErrorMessage
|
|
from helpers.io import WarningMessage
|
|
import Etesian
|
|
import Anabatic
|
|
import Katana
|
|
import Unicorn
|
|
import plugins
|
|
import clocktree.ClockTree
|
|
import chip.Configuration
|
|
import chip.BlockPower
|
|
import chip.BlockCorona
|
|
import chip.PadsCorona
|
|
except ImportError, e:
|
|
serror = str(e)
|
|
if serror.startswith('No module named'):
|
|
module = serror.split()[-1]
|
|
print '[ERROR] The <%s> python module or symbol cannot be loaded.' % module
|
|
print ' Please check the integrity of the <coriolis> package.'
|
|
if str(e).find('cannot open shared object file'):
|
|
library = serror.split(':')[0]
|
|
print '[ERROR] The <%s> shared library cannot be loaded.' % library
|
|
print ' Under RHEL 6, you must be under devtoolset-2.'
|
|
print ' (scl enable devtoolset-2 bash)'
|
|
sys.exit(1)
|
|
except Exception, e:
|
|
print '[ERROR] A strange exception occurred while loading the basic Coriolis/Python'
|
|
print ' modules. Something may be wrong at Python/C API level.\n'
|
|
print ' %s' % e
|
|
sys.exit(2)
|
|
|
|
|
|
# --------------------------------------------------------------------
|
|
# PlaceRoute class
|
|
|
|
|
|
class PlaceRoute ( object ):
|
|
|
|
def __init__ ( self, conf ):
|
|
self.conf = conf
|
|
self.validated = True
|
|
return
|
|
|
|
|
|
def _refresh ( self ):
|
|
if self.conf.viewer: self.conf.viewer.fit()
|
|
return
|
|
|
|
|
|
def validate ( self ):
|
|
self.validated = True
|
|
if len(self.conf.cores) < 1: self.validated = False
|
|
|
|
coreAb = self.conf.core.getAbutmentBox()
|
|
if (not coreAb.isEmpty()):
|
|
if coreAb.getWidth () <= self.conf.coreSize.getWidth() \
|
|
and coreAb.getHeight() <= self.conf.coreSize.getHeight():
|
|
self.conf.coreSize = coreAb
|
|
else:
|
|
raise ErrorMessage( 1, [ 'Core %s already have an abutment box, bigger than the requested one:'
|
|
% self.conf.cores[0].getName()
|
|
, " Cell abutment box: %s" % str(coreAb)
|
|
, " Maximum abutment box: %s" % str(self.conf.coreSize) ] )
|
|
self.validated = False
|
|
|
|
return self.validated
|
|
|
|
|
|
def doCoronaFloorplan ( self ):
|
|
if not self.validated:
|
|
raise ErrorMessage( 1, 'chip.doCoronaFloorplan(): Chip is not valid, aborting.' )
|
|
return
|
|
|
|
railsNb = Cfg.getParamInt('chip.block.rails.count' ).asInt()
|
|
hRailWidth = Cfg.getParamInt('chip.block.rails.hWidth' ).asInt()
|
|
vRailWidth = Cfg.getParamInt('chip.block.rails.vWidth' ).asInt()
|
|
hRailSpace = Cfg.getParamInt('chip.block.rails.hSpacing').asInt()
|
|
vRailSpace = Cfg.getParamInt('chip.block.rails.vSpacing').asInt()
|
|
|
|
if not self.conf.useClockTree: self.railsNb -= 1
|
|
|
|
innerBb = Box( self.conf.coreSize )
|
|
innerBb.inflate( (railsNb * vRailWidth + (railsNb+1) * vRailSpace + self.conf.getSliceHeight()) * 2
|
|
, (railsNb * hRailWidth + (railsNb+1) * hRailSpace + self.conf.getSliceHeight()) * 2 )
|
|
|
|
coronaAb = self.conf.corona.getAbutmentBox()
|
|
if innerBb.getWidth() > coronaAb.getWidth():
|
|
raise ErrorMessage( 1, 'Core is too wide to fit into the corona, needs %s but only has %s.'
|
|
% ( DbU.getValueString(innerBb .getWidth())
|
|
, DbU.getValueString(coronaAb.getWidth()) ) )
|
|
|
|
if innerBb.getHeight() > coronaAb.getHeight():
|
|
raise ErrorMessage( 1, 'Core is too tall to fit into the corona, needs %s but only has %s.'
|
|
% ( DbU.getValueString(innerBb .getHeight())
|
|
, DbU.getValueString(coronaAb.getHeight()) ) )
|
|
|
|
UpdateSession.open()
|
|
self.conf.core.setAbutmentBox( self.conf.coreSize )
|
|
x = (coronaAb.getWidth () - self.conf.coreSize.getWidth ()) / 2
|
|
y = (coronaAb.getHeight() - self.conf.coreSize.getHeight()) / 2
|
|
x = x - (x % self.conf.getSliceHeight())
|
|
y = y - (y % self.conf.getSliceHeight())
|
|
self.conf.icore.setTransformation ( Transformation(x,y,Transformation.Orientation.ID) )
|
|
self.conf.icore.setPlacementStatus( Instance.PlacementStatus.FIXED )
|
|
UpdateSession.close()
|
|
return
|
|
|
|
|
|
def doCorePlacement ( self ):
|
|
if not self.validated:
|
|
raise ErrorMessage( 1, 'chip.doCorePlacement(): Chip is not valid, aborting.' )
|
|
return
|
|
|
|
coreCell = self.conf.core
|
|
|
|
checkUnplaced = plugins.CheckUnplaced( coreCell, plugins.NoFlags )
|
|
if not checkUnplaced.check(): return
|
|
|
|
coreCk = None
|
|
if self.conf.coronaCk:
|
|
for plug in self.conf.coronaCk.getPlugs():
|
|
if plug.getInstance() == self.conf.icore:
|
|
coreCk = plug.getMasterNet()
|
|
if not coreCk:
|
|
print WarningMessage( 'Core <%s> is not connected to chip clock.' % self.conf.icore.getName() )
|
|
|
|
if self.conf.useClockTree and coreCk:
|
|
ht = clocktree.ClockTree.HTree.create( self.conf, coreCell, coreCk, coreCell.getAbutmentBox() )
|
|
ht.addCloned( self.conf.cell )
|
|
ht.addCloned( self.conf.corona )
|
|
etesian = Etesian.EtesianEngine.create( self.conf.corona )
|
|
etesian.setBlock( self.conf.icore )
|
|
etesian.setViewer( self.conf.viewer )
|
|
etesian.place()
|
|
etesian.destroy()
|
|
|
|
ht.connectLeaf()
|
|
ht.route()
|
|
ht.save( self.conf.cell )
|
|
else:
|
|
etesian = Etesian.EtesianEngine.create( self.conf.corona )
|
|
etesian.setBlock( self.conf.icore )
|
|
etesian.place()
|
|
etesian.destroy()
|
|
return
|
|
|
|
|
|
def doChipPlacement ( self ):
|
|
if not self.validated:
|
|
raise ErrorMessage( 1, 'chip.doChipPlacement(): Chip is not valid, aborting.' )
|
|
return
|
|
|
|
padsCorona = chip.PadsCorona.Corona( self.conf )
|
|
self.validated = padsCorona.validate()
|
|
if not self.validated: return False
|
|
|
|
padsCorona.doLayout()
|
|
self.validate()
|
|
self.doCoronaFloorplan()
|
|
self._refresh()
|
|
|
|
self.doCorePlacement()
|
|
self._refresh()
|
|
|
|
coreBlock = chip.BlockPower.Block( self.conf )
|
|
coreBlock.connectPower()
|
|
coreBlock.connectClock()
|
|
coreBlock.doLayout()
|
|
self._refresh()
|
|
|
|
coreCorona = chip.BlockCorona.Corona( coreBlock )
|
|
coreCorona.connectPads ( padsCorona )
|
|
coreCorona.connectBlock()
|
|
coreCorona.doLayout()
|
|
self._refresh()
|
|
|
|
return
|
|
|
|
|
|
def doChipRouting ( self ):
|
|
if not self.validated:
|
|
raise ErrorMessage( 1, 'chip.doChipRouting(): Chip is not valid, aborting.' )
|
|
return
|
|
|
|
self.conf.corona.setName( self.conf.corona.getName()+"_r" )
|
|
katana = Katana.KatanaEngine.create( self.conf.corona )
|
|
#katana.printConfiguration ()
|
|
katana.digitalInit ()
|
|
#katana.runNegociatePreRouted()
|
|
katana.runGlobalRouter ()
|
|
katana.loadGlobalRouting ( Anabatic.EngineLoadGrByNet )
|
|
katana.layerAssign ( Anabatic.EngineNoNetLayerAssign )
|
|
katana.runNegociate ( Katana.Flags.NoFlags )
|
|
success = katana.getToolSuccess()
|
|
katana.finalizeLayout()
|
|
katana.destroy()
|
|
|
|
return
|
|
|
|
|
|
def save ( self ):
|
|
if not self.validated:
|
|
raise ErrorMessage( 1, 'chip.save(): Chip is not valid, aborting.' )
|
|
return
|
|
|
|
af = CRL.AllianceFramework.get()
|
|
af.saveCell( self.conf.cell , CRL.Catalog.State.Views )
|
|
af.saveCell( self.conf.corona, CRL.Catalog.State.Views )
|
|
return
|