Python Interface for Hurricane / Coriolis

The (almost) complete interface of Hurricane is exported as a Python module and some parts of the other components of Coriolis (each one in a separate module). The interface has been made to mirror as closely as possible the C++ one, so the C++ doxygen documentation could be used to write code with either languages.

Summary of the C++ Documentation

A script could be run directly in text mode from the command line or through the graphical interface (see Executing Python Scripts in Cgt).

Aside for this requirement, the python script can contain anything valid in Python, so don’t hesitate to use any package or extension.

Small example of Python/Stratus script:

import symbolic.cmos
from   Hurricane import *
from   Stratus   import *

def doSomething ():
    # ...
    return

def ScriptMain ( **kw ):
  editor = None
  if kw.has_key('editor') and kw['editor']:
    editor = kw['editor']
    stratus.setEditor( editor )

  doSomething()
  return

if __name__ == "__main__" :
  kw           = {}
  success      = ScriptMain( **kw )
  shellSuccess = 0
  if not success: shellSuccess = 1

  sys.exit( shellSuccess )
      ScriptMain ()

This typical script can be executed in two ways:

  1. Run directly as a Python script, thanks to the

    if __name__ == "__main__" :
    

    part (this is standart Python). It is a simple adapter that will call ScriptMain().

    In this case, the import symbolic.cmos statement at the begining is mandatory.

  2. Through cgt, either in text or graphical mode. In that case, the ScriptMain() is directly called trough a sub-interpreter. The arguments of the script are passed through the **kw dictionnary.

    In this case, the import symbolic.cmos statement at the begining may be omitted.

    **kw Dictionnary
    Parameter Key/Name Contents type
    'cell' A Hurricane cell on which to work. Depending on the context, it may be None. For example, when run from cgt, the cell currently loaded in the viewer, if any.
    'editor' The viewer from which the script is run, when lauched through cgt.

Plugins

Plugins are Python scripts specially crafted to integrate with cgt. Their entry point is a ScriptMain() method as described in Python Interface to Coriolis. They can be called by user scripts through this method.

Chip Placement

Automatically performs the placement of a complete chip. This plugin, as well as the other P&R tools expect a specific top-level hierarchy for the design. The top-level hierarchy must contain the instances of all the I/O pads and exactly one instance named corona of an eponym cell corona. The corona cell in turn contains the instance of the chip’s core model.

The intermediate corona hierarchical level has been introduced to handle the possible decoupling between real I/O pads supplied by a foundry and a symbolic core. So the chip level contains only real layout and the corona and below only symbolic layer.

Note

This does not prevent having a design either fully symbolic (pads and core) or fully real.

Note

The corona also avoids the router to actually have to manage directly the pads which simplify its configuration and besides avoid to have the pads stuffed with blockages.

Chip Top Structure

The designer must provide a configuration file that defines the rules for the placement of the top-level hierarchy (that is, the pads and the core). This file must be names ioring.py and put into the user’s configuration directory ./coriolis2/

Example of chip placement configuration file (for AM2901):

from helpers import l, u, n

chip = \
  { 'pads.ioPadGauge' : 'pxlib'
  , 'pads.south'      : [ 'p_a3'     , 'p_a2'     , 'p_a1'     , 'p_r0'
                        , 'p_vddick0', 'p_vssick0', 'p_a0'     , 'p_i6'
                        , 'p_i8'     , 'p_i7'     , 'p_r3'     ]
  , 'pads.east'       : [ 'p_zero'   , 'p_i0'     , 'p_i1'     , 'p_i2'
                        , 'p_vddeck0', 'p_vsseck0', 'p_q3'     , 'p_b0'
                        , 'p_b1'     , 'p_b2'     , 'p_b3'     ]
  , 'pads.north'      : [ 'p_noe'    , 'p_y3'     , 'p_y2'     , 'p_y1'
                        , 'p_y0'     , 'p_vddeck1', 'p_vsseck1', 'p_np'
                        , 'p_ovr'    , 'p_cout'   , 'p_ng'     ]
  , 'pads.west'       : [ 'p_cin'    , 'p_i4'     , 'p_i5'     , 'p_i3'
                        , 'p_ck'     , 'p_d0'     , 'p_d1'     , 'p_d2'
                        , 'p_d3'     , 'p_q0'     , 'p_f3'     ]
  , 'core.size'       : ( l(1500), l(1500) )
  , 'chip.size'       : ( l(3000), l(3000) )
  , 'chip.clockTree'  : True
  }

The file must contain one dictionnary named chip.

Chip Dictionnary
Parameter Key/Name Value/Contents type
'pad.ioPadGauge' The routing gauge to use for the pad. Must be given as it differs from the one used to route inside the core
'pad.south' Ordered list (left to right) of pad instance names to put on the south side of the chip
'pad.east' Ordered list (down to up) of pad instance names to put on the east side of the chip
'pad.north' Ordered list (left to right) of pad instance names to put on the north side of the chip
'pad.west' Ordered list (down to up) of pad instance names to put on the west side of the chip
'core.size' The size of the core (to be used by the placer)
'chip.size' The size of the whole chip. The sides must be large enough to accomodate all the pads
'chip.clockTree' Whether to generate a clock tree or not. This calls the ClockTree plugin

Configuration parameters, defaults are defined in etc/coriolis2/<STECHNO>/plugins.conf.

Parameter Identifier Type Default
Chip Plugin Parameters
chip.block.rails.count Int 5
The minimum number of rails around the core block. Must be odd and above 5. One rail for the clock and at least two pairs of power/grounds
chip.block.rails.hWidth Int 12 λ
The horizontal width of the rails
chip.block.rails.vWidth Int 12 λ
The vertical width of the rails
chip.block.rails.hSpacing Int 6 λ
The spacing, edge to edge of two adjacent horizontal rails
chip.block.rails.vSpacing Int 6 λ
The spacing, edge to edge of two adjacent vertical rails

Note

If no clock tree is generated, then the clock rail is not created. So even if the requested number of rails chip.block.rails.count is, say 5, only four rails (2* power, 2* ground) will be generated.

Clock Tree

Inserts a clock tree into a block. The clock tree uses the H strategy. The clock net is splitted into sub-nets, one for each branch of the tree.

  • On chip design, the sub-nets are created in the model of the core block (then trans-hierarchically flattened to be shown at chip level).

  • On blocks, the sub nets are created directly in the top block.

  • The sub-nets are named according to a simple geometrical scheme. A common prefix ck_htree, then one postfix by level telling on which quarter of plane the sub-clock is located:

    1. _bl: bottom left plane quarter.
    2. _br: bottom right plane quarter.
    3. _tl: top left plane quarter.
    4. _tr: top right plane quarter.

    We can have ck_htree_bl, ck_htree_bl_bl, ch_htree_bl_tl and so on.

The clock tree plugin works in four steps:

  1. Builds the clock tree: creates the top-block abutment box, computes the required levels of H tree and places the clock buffers.
  2. Once the clock buffers are placed, calls the placer (Etesian) to place the ordinary standard cells, whithout disturbing clock H-tree buffers.
  3. At this point we know the exact positions of all the DFFs, so we can connect them to the nearest H-tree leaf clock signal.
  4. Leaf clock signals that are not connected to any DFFs are removed.

Netlist reorganisation:

  • Obviously the top block or chip core model netlist is modified to contain all the clock sub-nets. The interface is not changed.
  • If the top block contains instances of other models and those models contain DFFs that get re-connected to the clock sub-nets (from the top level): Changes both the model netlist and interface to propagate the relevant clock sub-nets to the instanciated model. The new model with the added clock signal is renamed with a _cts suffix. For example, the sub-block model ram.vst will become ram_cts.vst.

Note

If you are to re-run the clock tree plugin on a netlist, be careful to erase any previously generated _cts file (both netlist and layout: rm *_cts.{ap,vst}). And restart cgt to clear its memory cache.

Configuration parameters, defaults are defined in etc/coriolis2/<STECHNO>/plugins.conf.

Parameter Identifier Type Default
ClockTree Plugin Parameters
clockTree.minimumSide Int 300 λ
The minimum size below which the clock tree will stop to perform quadri-partitions
clockTree.buffer String buf_x2
The buffer model to use to drive sub-nets

Recursive-Save (RSave)

Performs a recursive top down save of all the models from the top cell loaded in cgt. Forces a write of any non-terminal model. This plugin is used by the clock tree plugin after the netlist clock sub-nets creation.

A Simple Example: AM2901

To illustrate the capabilities of Coriolis tools and Python scripting, a small example, derived from the Alliance AM2901 is supplied.

This example contains only the synthetized netlists and the doChip.py script which perform the whole P&R of the design.

You can generate the chip using one of the following methods:

  1. Command line mode: directly run the script:

    dummy@lepka:AM2901> ./doChip -V --cell=amd2901
    
  2. Graphic mode: launch cgt, load chip netlist amd2901 (the top cell) then run the Python script doChip.py.

Note

Between two consecutive run, be sure to erase the netlist/layout generated:

dummy@lepka:AM2901> rm *_cts*.vst *.ap