# -*- mode:Python -*-
#
# This file is part of the Coriolis Software.
# Copyright (c) Sorbonne Université 2020-2021, All Rights Reserved
#
# +-----------------------------------------------------------------+ 
# |                   C O R I O L I S                               |
# |          Alliance / Hurricane  Interface                        |
# |                                                                 |
# |  Author      :                    Jean-Paul Chaput              |
# |  E-mail      :            Jean-Paul.Chaput@lip6.fr              |
# | =============================================================== |
# |  Python      :   "./crlcore/helpers/technology.py"              |
# +-----------------------------------------------------------------+

"""
Some helpers to create or load a technology and it's libraries.
"""

from Hurricane3  import DataBase, Layer, BasicLayer, ViaLayer


__all__ = [ 'safeGetLibrary', 'createBL', 'setEnclosures' ]


def safeGetLibrary ( frameworkName, libName ):
    """
    Return the library named ``libName`` if it already exists,
    or create it if needed. To avoid potential collision, design
    kits libraries should not be created directly under the DataBase
    root library but be grouped under a common technology library,
    in this case ``CM018``.
    """
    db       = DataBase.getDB()
    tech     = db.getTechnology()
    frameLib = db.getRootLibrary().getLibrary( frameworkName )
    if frameLib is None:
        frameLib = Library.create( db.getRootLibrary(), frameworkName )
    lib = frameLib.getLibrary( libName )
    if lib is None:
        lib = Library.create( frameLib, libName )
    return lib


def createBL ( tech, layerName, material, size=None, spacing=None, gds2Layer=None, gds2DataType=0, area=None ):
    """
    Create a new BasicLayer. Parameters ``tech``, ``layerName`` and ``material``
    are mandatory.

    :param tech:         The technology the basic layer will be part of.
    :param layerName:    The name of the layer.
    :param material:     The kind of layer, see BasicLayer.Material.
    :param size:         The minimal size (i.e. width).
    :param spacing:      The minimal distance, edge to edge between two wires.
    :param gds2layer:    The GDSII layer number (for the GDSII driver).
    :param gds2DataType: The GDSII DataType (i.e purpose).
    :param area:         The minimum area (in um2)
    """
    print( tech )
    layer = BasicLayer.create( tech, layerName, BasicLayer.Material(material) )
    if size is not None:
        layer.setMinimalSize( size )
    if spacing is not None:
        layer.setMinimalSpacing( spacing )
    if gds2Layer is not None:
        layer.setGds2Layer   ( gds2Layer )
        layer.setGds2Datatype( gds2DataType )
    if area is not None:
        layer.setMinimalArea( area )
    return layer


def createVia ( tech, viaName, botName, cutName, topName, size=None ):
    """
    Create a new ViaLayer. Parameters ``tech``, ``viaName``, ``botName``,
    ``cutName`` and ``topName`` are mandatory.

    :param tech:    The technology the basic layer will be part of.
    :param viaName: The name of the newly defined VIA layer.
    :param botName: The name of the *bottom* metal layer.
    :param cutName: The name of the *cut* (aka, via hole) layer.
    :param topName: The name of the *top* metal layer.
    :param size:    The minimal side size of the square *cut* layer.
    """
    layer = ViaLayer.create( tech
                           , viaName
                           , tech.getLayer(botName)
                           , tech.getLayer(cutName)
                           , tech.getLayer(topName) )
    if size is not None:
        layer.setMinimalSize( size )
    return layer


def setEnclosures ( layer, subLayer, enclosures ):
    """
    Set horizontal & vertical enclosure for a given ``subLayer`` in a
    composite ``layer`` (typically a ViaLayer). If ``enclosures`` is a
    number, both H/V will be set to that same value. If it is a tuple
    (a pair), then the first value is horizontal and the seconf is
    vertical.
    """
    if isinstance(enclosures,tuple):
       henclosure = enclosures[0]
       venclosure = enclosures[1]
    else:
       henclosure = enclosures
       venclosure = enclosures
    layer.setEnclosure( subLayer, henclosure, Layer.EnclosureH )
    layer.setEnclosure( subLayer, venclosure, Layer.EnclosureV )
    return