• Skip to main content
  • Skip to search
  • Skip to footer
Cadence Home
  • This search text may be transcribed, used, stored, or accessed by our third-party service providers per our Cookie Policy and Privacy Policy.

  1. Community Forums
  2. Custom IC SKILL
  3. How to create a transistor pcell in SKILL?

Stats

  • Locked Locked
  • Replies 13
  • Subscribers 145
  • Views 24608
  • Members are here 0
This discussion has been locked.
You can no longer post new replies to this discussion. If you have a question you can start a new discussion

How to create a transistor pcell in SKILL?

Elezebeth
Elezebeth over 16 years ago

Hi,

I am new to SKILL coding.I have not done any pcell coding in SKILL.I would like to know whether we can create mosfet pcells using SKILL code.The way I tried to do this is using rodCreateRect( ) to create different layers of the transistor say diffusion,poly,contacts etc.and allign all these layers using rodAlign( ).But the transistor that got created could not be deleted at a time....ie.the layout is flattened.If i select the rectangle that I have set as reference and try to delete the whole transistor layout doesn't get deleted, only that particular layer is deleted.

The other way I found is to do using ?subRectArray.but the problem with that is, the number of contacts in a transistor will increase as the width of the transistor increase.So the number of sub-rectangles that have to be created is variable.How to do that?

Can anybody please guide me on this?

Thanks and Regards,

Elezebeth 

 

 

  • Cancel
Parents
  • Andrew Beckett
    Andrew Beckett over 16 years ago

    It sounds as if you're not actually creating a pcell, but just creating level-0 ROD objects.

    Here's a simple ROD-based example that might get you started:

    ;
    ; This pcell demonstrates the use of sub-rectangles and offset subpaths
    ; in the multipart path transistor implementation. The name of the master
    ; view is:
    ;
    ; pcells step8 layout
    ;
    ; This pcell accepts these parameters:
    ;
    ; width Width of the transistor's active area (note: this is a
    ; distance measured in the Y-direction).
    ; (float, default = 3.0)
    ;
    ; length Length of the transistor's active area (note: this is a
    ; distance measured in the X-direction).
    ; (float, default = 0.6)
    ;
    ; polyLayer Name of the poly layer.
    ; (string, default = "poly")
    ;
    ; diffLayer Name of the diffusion layer.
    ; (string, default = "pdiff")
    ;
    ; contLayer Name of the contact layer.
    ; (string, default = "cont")
    ;
    ; metalLayer Name of the metal layer.
    ; (string, default = "metal1")
    ;
    ; drainName Name of the drain connection.
    ; (string, default = "D")
    ;
    ; gateName Name of the gate connection.
    ; (string, default = "G")
    ;
    ; sourceName Name of the source connection.
    ; (string, default = "S")
    ;
    ; sourceContact Whether source contacts should be included
    ; (boolean, default = t)
    ;
    ; drainContact Whether drain contacts should be included
    ; (boolean, default = t)
    ;
    pcDefinePCell(
    ; Identify the target cellview.
    list(ddGetObj("pcells") "step8" "layout")

    ; Define formal parameter name-value pairs.
    (
    (width 3.0)
    (length 0.6)
    (polyLayer "poly")
    (diffLayer "pdiff")
    (contLayer "cont")
    (metalLayer "metal1")
    (drainName "D")
    (gateName "G")
    (sourceName "S")
    (sourceContact t)
    (drainContact t)
    )

    ; Define the contents of this cellview.
    let((tfId polyExtend contWidth polyContSep diffContEnclose
    metalContEnclose diffPolyEnclose pinEndOffset polySep transObj grid
    polyDiffSep sdWidth)

    ; convert to proper booleans
    sourceContact=(sourceContact==t || sourceContact=="TRUE")
    drainContact=(drainContact==t || drainContact=="TRUE")

    ; Get the technology information for this cell.
    tfId = techGetTechFile(ddGetObj("pcells"))

    ; Get the minimum extension of poly beyond diffusion.
    polyExtend = techGetSpacingRule(tfId "minExtension" polyLayer)

    ; Get the minimum contact width.
    contWidth = techGetSpacingRule(tfId "minWidth" contLayer)

    ; Get the gate spacing
    polySep = techGetSpacingRule(tfId "minSpacing" polyLayer)

    ; Get the minimum poly to contact spacing.
    polyContSep = techGetSpacingRule(tfId "minSpacing" polyLayer contLayer)

    ; Get the minimum poly to diffusion spacing.
    polyDiffSep = techGetSpacingRule(tfId "minSpacing" polyLayer diffLayer)

    ; Get the minimum diffusion enclosure of contact.
    diffContEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" diffLayer contLayer)

    ; Get the minimum diffusion enclosure of poly (used when metal contacts
    ; not included)
    diffPolyEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" diffLayer polyLayer)

    ; Get the minimum metal enclosure of contact.
    metalContEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" metalLayer contLayer)

    ; pin end offset
    pinEndOffset = -(polyExtend + diffContEnclose - metalContEnclose)

    ; source/drain width
    sdWidth = contWidth+polyContSep+diffContEnclose

    ; Get the minimum resolution for this process.
    grid = techGetMfgGridResolution(tfId)

    ;------------------------------------------------------------------
    ; Create the gate and diffusion regions.
    ;------------------------------------------------------------------
    transObj = rodCreatePath(
    ?layer list(polyLayer "drawing")
    ?endType "variable"
    ?width length
    ?pts list(0.0:0.0 0.0:width)
    ?beginExt polyExtend
    ?endExt polyExtend
    ; Define the diffusion region - just under the gate
    ?encSubPath
    setof(arg
    list(
    list(
    ?layer list(diffLayer "drawing")
    ?enclosure 0.0
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    )
    arg
    )
    ?offsetSubPath
    setof(arg
    list(
    ; Define the drain metal stripe.
    when(drainContact
    list(
    ?layer list(metalLayer "drawing")
    ?pin t
    ?termName drainName
    ?justification "left"
    ?width contWidth + 2.0*metalContEnclose
    ?sep polyContSep - metalContEnclose
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    ; Define the source metal stripe.
    when(sourceContact
    list(
    ?layer list(metalLayer "drawing")
    ?pin t
    ?termName sourceName
    ?justification "right"
    ?width contWidth + 2.0*metalContEnclose
    ?sep polyContSep - metalContEnclose
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    ; Define the diffusion source area
    list(
    ?layer list(diffLayer "drawing")
    ; width depends on whether source contact included or not
    ?width if(sourceContact sdWidth diffPolyEnclose)
    ?justification "right"
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    ; Define the diffusion drain area
    list(
    ?layer list(diffLayer "drawing")
    ; width depends on whether drain contact included or not
    ?width if(drainContact sdWidth diffPolyEnclose)
    ?justification "left"
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    ; Define the drain diffusion pin. For abut reasons the pin is
    ; the same height that the metal pin is
    ; NOTE - if the difference between diffPolyEnclose and polySep
    ; is greater than the minimum gate length, then when abutted
    ; the drain pin from one transistor can short out the
    ; other transistor. This could be dealt with if necessary by
    ; making the code a bit smarter
    list(
    ?layer list(diffLayer "pin")
    ?pin t
    ?termName drainName
    ?justification "left"
    ; width adjusts to ensure that the diffusion pins will touch
    ; when the devices abut
    ?width if(drainContact sdWidth diffPolyEnclose)
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    ; Define the source diffusion pin. For abut reasons the pin is
    ; the same height that the metal pin is
    list(
    ?layer list(diffLayer "pin")
    ?pin t
    ?termName sourceName
    ?justification "right"
    ?width if(sourceContact sdWidth diffPolyEnclose)
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    arg
    )
    ?subRect
    setof(arg
    list(
    ; Define the drain contact array.
    when(drainContact
    list(
    ?layer list(contLayer "drawing")
    ?justification "left"
    ?sep polyContSep
    ?beginOffset -(polyExtend + diffContEnclose)
    ?endOffset -(polyExtend + diffContEnclose)
    )
    )
    ; Define the source contact array.
    when(sourceContact
    list(
    ?layer list(contLayer "drawing")
    ?justification "right"
    ?sep polyContSep
    ?beginOffset -(polyExtend + diffContEnclose)
    ?endOffset -(polyExtend + diffContEnclose)
    )
    )
    ; Define the gate pins.
    list(
    ?layer list(polyLayer "pin")
    ?pin t
    ?termName gateName
    ?width length
    ?length grid
    ?space width - 2*grid
    )
    )
    arg
    )
    )
    rodCreateRect(
    ?layer contLayer
    ?width 2.0
    ?length 3.0
    )
    ; rodUnNameShape(transObj)

    ;------------------------------------------------------------------
    ; Dimensions for auto-abutment
    ; This is fairly simplistic, and in the case of different sized
    ; devices it can end up with the spacing between gates not quite
    ; as tight as possible in some cases. However, it demonstrates
    ; the principle.
    ; These dimensions are used in the abut function
    ;------------------------------------------------------------------
    pcCellView~>sameWidthDist=-diffPolyEnclose
    pcCellView~>diffWidthDist=polyDiffSep-diffPolyEnclose

    ;------------------------------------------------------------------
    ; Prevent overlap markers
    ;------------------------------------------------------------------
    pcCellView~>lxBlockOverlapCheck=t

    ;------------------------------------------------------------------
    ; Add auto-abutment properties
    ;------------------------------------------------------------------
    foreach(pin dbFindTermByName(pcCellView "S")~>pins
    pin~>fig~>abutFunction="abAbutFunction"
    pin~>fig~>abutAccessDir='("right")
    )
    foreach(pin dbFindTermByName(pcCellView "D")~>pins
    pin~>fig~>abutFunction="abAbutFunction"
    pin~>fig~>abutAccessDir='("left")
    )

    )
    )

    Regards,

    Andrew.

    • Cancel
    • Vote Up +1 Vote Down
    • Cancel
Reply
  • Andrew Beckett
    Andrew Beckett over 16 years ago

    It sounds as if you're not actually creating a pcell, but just creating level-0 ROD objects.

    Here's a simple ROD-based example that might get you started:

    ;
    ; This pcell demonstrates the use of sub-rectangles and offset subpaths
    ; in the multipart path transistor implementation. The name of the master
    ; view is:
    ;
    ; pcells step8 layout
    ;
    ; This pcell accepts these parameters:
    ;
    ; width Width of the transistor's active area (note: this is a
    ; distance measured in the Y-direction).
    ; (float, default = 3.0)
    ;
    ; length Length of the transistor's active area (note: this is a
    ; distance measured in the X-direction).
    ; (float, default = 0.6)
    ;
    ; polyLayer Name of the poly layer.
    ; (string, default = "poly")
    ;
    ; diffLayer Name of the diffusion layer.
    ; (string, default = "pdiff")
    ;
    ; contLayer Name of the contact layer.
    ; (string, default = "cont")
    ;
    ; metalLayer Name of the metal layer.
    ; (string, default = "metal1")
    ;
    ; drainName Name of the drain connection.
    ; (string, default = "D")
    ;
    ; gateName Name of the gate connection.
    ; (string, default = "G")
    ;
    ; sourceName Name of the source connection.
    ; (string, default = "S")
    ;
    ; sourceContact Whether source contacts should be included
    ; (boolean, default = t)
    ;
    ; drainContact Whether drain contacts should be included
    ; (boolean, default = t)
    ;
    pcDefinePCell(
    ; Identify the target cellview.
    list(ddGetObj("pcells") "step8" "layout")

    ; Define formal parameter name-value pairs.
    (
    (width 3.0)
    (length 0.6)
    (polyLayer "poly")
    (diffLayer "pdiff")
    (contLayer "cont")
    (metalLayer "metal1")
    (drainName "D")
    (gateName "G")
    (sourceName "S")
    (sourceContact t)
    (drainContact t)
    )

    ; Define the contents of this cellview.
    let((tfId polyExtend contWidth polyContSep diffContEnclose
    metalContEnclose diffPolyEnclose pinEndOffset polySep transObj grid
    polyDiffSep sdWidth)

    ; convert to proper booleans
    sourceContact=(sourceContact==t || sourceContact=="TRUE")
    drainContact=(drainContact==t || drainContact=="TRUE")

    ; Get the technology information for this cell.
    tfId = techGetTechFile(ddGetObj("pcells"))

    ; Get the minimum extension of poly beyond diffusion.
    polyExtend = techGetSpacingRule(tfId "minExtension" polyLayer)

    ; Get the minimum contact width.
    contWidth = techGetSpacingRule(tfId "minWidth" contLayer)

    ; Get the gate spacing
    polySep = techGetSpacingRule(tfId "minSpacing" polyLayer)

    ; Get the minimum poly to contact spacing.
    polyContSep = techGetSpacingRule(tfId "minSpacing" polyLayer contLayer)

    ; Get the minimum poly to diffusion spacing.
    polyDiffSep = techGetSpacingRule(tfId "minSpacing" polyLayer diffLayer)

    ; Get the minimum diffusion enclosure of contact.
    diffContEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" diffLayer contLayer)

    ; Get the minimum diffusion enclosure of poly (used when metal contacts
    ; not included)
    diffPolyEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" diffLayer polyLayer)

    ; Get the minimum metal enclosure of contact.
    metalContEnclose =
    techGetOrderedSpacingRule(tfId "minEnclosure" metalLayer contLayer)

    ; pin end offset
    pinEndOffset = -(polyExtend + diffContEnclose - metalContEnclose)

    ; source/drain width
    sdWidth = contWidth+polyContSep+diffContEnclose

    ; Get the minimum resolution for this process.
    grid = techGetMfgGridResolution(tfId)

    ;------------------------------------------------------------------
    ; Create the gate and diffusion regions.
    ;------------------------------------------------------------------
    transObj = rodCreatePath(
    ?layer list(polyLayer "drawing")
    ?endType "variable"
    ?width length
    ?pts list(0.0:0.0 0.0:width)
    ?beginExt polyExtend
    ?endExt polyExtend
    ; Define the diffusion region - just under the gate
    ?encSubPath
    setof(arg
    list(
    list(
    ?layer list(diffLayer "drawing")
    ?enclosure 0.0
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    )
    arg
    )
    ?offsetSubPath
    setof(arg
    list(
    ; Define the drain metal stripe.
    when(drainContact
    list(
    ?layer list(metalLayer "drawing")
    ?pin t
    ?termName drainName
    ?justification "left"
    ?width contWidth + 2.0*metalContEnclose
    ?sep polyContSep - metalContEnclose
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    ; Define the source metal stripe.
    when(sourceContact
    list(
    ?layer list(metalLayer "drawing")
    ?pin t
    ?termName sourceName
    ?justification "right"
    ?width contWidth + 2.0*metalContEnclose
    ?sep polyContSep - metalContEnclose
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    ; Define the diffusion source area
    list(
    ?layer list(diffLayer "drawing")
    ; width depends on whether source contact included or not
    ?width if(sourceContact sdWidth diffPolyEnclose)
    ?justification "right"
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    ; Define the diffusion drain area
    list(
    ?layer list(diffLayer "drawing")
    ; width depends on whether drain contact included or not
    ?width if(drainContact sdWidth diffPolyEnclose)
    ?justification "left"
    ?beginOffset -polyExtend
    ?endOffset -polyExtend
    )
    ; Define the drain diffusion pin. For abut reasons the pin is
    ; the same height that the metal pin is
    ; NOTE - if the difference between diffPolyEnclose and polySep
    ; is greater than the minimum gate length, then when abutted
    ; the drain pin from one transistor can short out the
    ; other transistor. This could be dealt with if necessary by
    ; making the code a bit smarter
    list(
    ?layer list(diffLayer "pin")
    ?pin t
    ?termName drainName
    ?justification "left"
    ; width adjusts to ensure that the diffusion pins will touch
    ; when the devices abut
    ?width if(drainContact sdWidth diffPolyEnclose)
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    ; Define the source diffusion pin. For abut reasons the pin is
    ; the same height that the metal pin is
    list(
    ?layer list(diffLayer "pin")
    ?pin t
    ?termName sourceName
    ?justification "right"
    ?width if(sourceContact sdWidth diffPolyEnclose)
    ?beginOffset pinEndOffset
    ?endOffset pinEndOffset
    )
    )
    arg
    )
    ?subRect
    setof(arg
    list(
    ; Define the drain contact array.
    when(drainContact
    list(
    ?layer list(contLayer "drawing")
    ?justification "left"
    ?sep polyContSep
    ?beginOffset -(polyExtend + diffContEnclose)
    ?endOffset -(polyExtend + diffContEnclose)
    )
    )
    ; Define the source contact array.
    when(sourceContact
    list(
    ?layer list(contLayer "drawing")
    ?justification "right"
    ?sep polyContSep
    ?beginOffset -(polyExtend + diffContEnclose)
    ?endOffset -(polyExtend + diffContEnclose)
    )
    )
    ; Define the gate pins.
    list(
    ?layer list(polyLayer "pin")
    ?pin t
    ?termName gateName
    ?width length
    ?length grid
    ?space width - 2*grid
    )
    )
    arg
    )
    )
    rodCreateRect(
    ?layer contLayer
    ?width 2.0
    ?length 3.0
    )
    ; rodUnNameShape(transObj)

    ;------------------------------------------------------------------
    ; Dimensions for auto-abutment
    ; This is fairly simplistic, and in the case of different sized
    ; devices it can end up with the spacing between gates not quite
    ; as tight as possible in some cases. However, it demonstrates
    ; the principle.
    ; These dimensions are used in the abut function
    ;------------------------------------------------------------------
    pcCellView~>sameWidthDist=-diffPolyEnclose
    pcCellView~>diffWidthDist=polyDiffSep-diffPolyEnclose

    ;------------------------------------------------------------------
    ; Prevent overlap markers
    ;------------------------------------------------------------------
    pcCellView~>lxBlockOverlapCheck=t

    ;------------------------------------------------------------------
    ; Add auto-abutment properties
    ;------------------------------------------------------------------
    foreach(pin dbFindTermByName(pcCellView "S")~>pins
    pin~>fig~>abutFunction="abAbutFunction"
    pin~>fig~>abutAccessDir='("right")
    )
    foreach(pin dbFindTermByName(pcCellView "D")~>pins
    pin~>fig~>abutFunction="abAbutFunction"
    pin~>fig~>abutAccessDir='("left")
    )

    )
    )

    Regards,

    Andrew.

    • Cancel
    • Vote Up +1 Vote Down
    • Cancel
Children
  • youcanred
    youcanred over 6 years ago in reply to Andrew Beckett

    Hi Andrew, I tried this transistor pcell code. changed the library name to my library name. But it has

    *error* techGetSpacingRule: argument #1 should be a database object (type template = "dtgg") - nil. 

    Any idea what kind of error it is ? i tried this code in IC6.1.5

    *Warning* (Tech-2000010): techGetTechFile: Argument must be dbCellViewId or ddId for cell,library,and so forth -nil

    thank you so much!

    regards,

    Jocelyn

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel
  • Andrew Beckett
    Andrew Beckett over 6 years ago in reply to youcanred

    Jocelyn,

    You would need to change two lines:

       list(ddGetObj("pcells") "step8" "layout")

    and

          ;  Get the technology information for this cell.
    tfId = techGetTechFile(ddGetObj("pcells"))

    to point to the library you want to create the pcell in - and assuming this is attached (or referencing) the correct technology library. You may also need to edit the layer names in the code too. It will be the second in particular that is causing the errors you're seeing - if there's no library called "pcells" then ddGetObj would return nil, hence the error you are getting.

    Regards,

    Andrew.

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel
  • youcanred
    youcanred over 6 years ago in reply to Andrew Beckett

    Hi Andrew,

    Thank you, i did forget to change the second "pcells".  However, i changed both two parts to my libraray name, it appears another error.

    *Error* drain: port cannot be drained - port:"/tmp/pcGenqc8741"

    Regards,

    Jocelyn

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel
  • Andrew Beckett
    Andrew Beckett over 6 years ago in reply to youcanred

    Hi Jocelyn,

    That is a temporary file used during the pcDefinePCell call. It sounds as if /tmp has run out of disk space - you should check that and maybe clear up some space (lots of things are likely to go wrong in the operating system if /tmp is full anyway).

    Regards,

    Andrew.

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel
  • MOHANRAJ A
    MOHANRAJ A over 5 years ago in reply to Andrew Beckett

    Hi Andrew, I tried this transistor pcell code. changed the library name to my library name on both sides,

    But i am getting below warning message,Any idea what kind of error it is ? i tried this code in IC6.1.6

    *WARNING* (TECH-290096): techGetSpacingRule: Layer-purpose pair ("poly") was not defined.
    *WARNING* (TECH-290096): techGetSpacingRule: Layer-purpose pair ("cont") was not defined.
    *WARNING* (TECH-290096): techGetSpacingRule: Layer-purpose pair ("poly") was not defined.
    *WARNING* (TECH-290096): techGetSpacingRule: Layer-purpose pair ("poly") was not defined.
    *WARNING* (TECH-290096): techGetSpacingRule: Layer-purpose pair ("poly") was not defined.
    *WARNING* (TECH-290096): techGetOrderedSpacingRule: Layer-purpose pair ("cont") was not defined.
    *WARNING* (TECH-290096): techGetOrderedSpacingRule: Layer-purpose pair ("poly") was not defined.
    *WARNING* (TECH-290096): techGetOrderedSpacingRule: Layer-purpose pair ("metal1") was not defined.

    Thanks you

    Regards,

    Mohan

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel
  • Andrew Beckett
    Andrew Beckett over 5 years ago in reply to MOHANRAJ A

    Hi Mohan,

    As I said in my earlier reply (10 months ago) to this old thread from 11 years ago...

    Andrew Beckett said:
    You may also need to edit the layer names in the code too.

    So you'd need to change:

          (polyLayer "poly")
    (diffLayer "pdiff")
    (contLayer "cont")
    (metalLayer "metal1")

    to relevant layer names for your technology.

    Andrew.

    • Cancel
    • Vote Up 0 Vote Down
    • Cancel

Community Guidelines

The Cadence Design Communities support Cadence users and technologists interacting to exchange ideas, news, technical information, and best practices to solve problems and get the most from Cadence technology. The community is open to everyone, and to provide the most value, we require participants to follow our Community Guidelines that facilitate a quality exchange of ideas and information. By accessing, contributing, using or downloading any materials from the site, you agree to be bound by the full Community Guidelines.

© 2025 Cadence Design Systems, Inc. All Rights Reserved.

  • Terms of Use
  • Privacy
  • Cookie Policy
  • US Trademarks
  • Do Not Sell or Share My Personal Information