diff --git a/kpex/kpex_cli.py b/kpex/kpex_cli.py index 66b7081f..d0544850 100755 --- a/kpex/kpex_cli.py +++ b/kpex/kpex_cli.py @@ -87,10 +87,10 @@ class KpexCLI: def parse_args(arg_list: List[str] = None) -> argparse.Namespace: # epilog = f"See '{PROGRAM_NAME} -h' for help on subcommand" epilog = """ -| Variable | Example | -| -------- | -------------------- | -| PDKPATH | (e.g. $HOME/.volare) | -| PDK | (e.g. sky130A) | +| Variable | Example | Description | +| -------- | -------------------- | --------------------------------------- | +| PDKPATH | (e.g. $HOME/.volare) | Optional (required for default magigrc) | +| PDK | (e.g. sky130A) | Optional (required for default magigrc) | """ epilog_md = rich.console.Group( rich.text.Text('Environmental variables:', style='argparse.groups'), @@ -128,7 +128,8 @@ def parse_args(arg_list: List[str] = None) -> argparse.Namespace: group_pex_input.add_argument("--lvsdb", "-l", dest="lvsdb_path", help="KLayout LVSDB path (bypass LVS)") group_pex_input.add_argument("--cell", "-c", dest="cell_name", default=None, help="Cell (default is the top cell)") - default_lvs_script_path = os.path.realpath(os.path.join(__file__, '..', '..', 'pdk', 'sky130A', 'kpex', 'sky130.lvs')) + default_lvs_script_path = os.path.realpath(os.path.join(__file__, '..', '..', 'pdk', + 'sky130A', 'libs.tech', 'kpex', 'sky130.lvs')) group_pex_input.add_argument("--lvs_script", dest="lvs_script_path", default=default_lvs_script_path, @@ -199,7 +200,10 @@ def parse_args(arg_list: List[str] = None) -> argparse.Namespace: action='store_true', default=False, help="FasterCap -pj Use Jacobi preconditioner (default is %(default)s)") - default_magicrc_path = os.path.abspath(f"{os.environ['PDKPATH']}/libs.tech/magic/{os.environ['PDK']}.magicrc") + PDKPATH = os.environ.get('PDKPATH', None) + default_magicrc_path = \ + None if PDKPATH is None \ + else os.path.abspath(f"{PDKPATH}/libs.tech/magic/{os.environ['PDK']}.magicrc") group_magic = main_parser.add_argument_group("MAGIC options") group_magic.add_argument('--magicrc', dest='magicrc_path', default=default_magicrc_path, help=f"Path to magicrc configuration file (default is '%(default)s')") diff --git a/pdk/ihp_sg13g2/kpex/rule_decks b/pdk/ihp_sg13g2/kpex/rule_decks deleted file mode 120000 index 04451aaa..00000000 --- a/pdk/ihp_sg13g2/kpex/rule_decks +++ /dev/null @@ -1 +0,0 @@ -../../../../IHP-Open-PDK/ihp-sg13g2/libs.tech/klayout/tech/lvs/rule_decks \ No newline at end of file diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_connections.lvs new file mode 100644 index 00000000..a1cc5611 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_connections.lvs @@ -0,0 +1,69 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------ BJT CONNECTIONS -------- +#================================ + +logger.info('Starting LVS BJT CONNECTIONS') + +# ============= +# ---- NPN ---- +# ============= + +# General +connect(npn_sub, pwell) + +# npn13G2 nodes connections +connect(npn13G2_te, npn13G2_e_pin) +connect(npn13G2_tc, npn13G2_c_pin) +connect(npn13G2_tb, npn13G2_b_pin) +connect(npn13G2_e_pin, cont_drw) +connect(npn13G2_c_pin, cont_drw) +connect(npn13G2_b_pin, cont_drw) +connect(npn_sub, pwell) +connect(npn13G2_e_pin, emwind_drw) +connect(emwind_drw, metal1_con) + +# npn13G2L nodes connections +connect(npn13G2l_te, npn13G2l_e_pin) +connect(npn13G2l_tc, npn13G2l_c_pin) +connect(npn13G2l_tb, npn13G2l_b_pin) +connect(npn13G2l_e_pin, cont_drw) +connect(npn13G2l_c_pin, cont_drw) +connect(npn13G2l_b_pin, cont_drw) +connect(npn_sub, pwell) + +# npn13G2V nodes connections +connect(npn13G2v_te, npn13G2v_e_pin) +connect(npn13G2v_tc, npn13G2v_c_pin) +connect(npn13G2v_tb, npn13G2v_b_pin) +connect(npn13G2v_e_pin, cont_drw) +connect(npn13G2v_c_pin, cont_drw) +connect(npn13G2v_b_pin, cont_drw) +connect(npn_sub, pwell) + +# ============= +# ---- PNP ---- +# ============= + +# pnp_mpa nodes connections +connect(pnp_mpa_e, cont_drw) +connect(pnp_mpa_b, cont_drw) +connect(pnp_mpa_c, cont_drw) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_derivations.lvs new file mode 100644 index 00000000..0d56b899 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_derivations.lvs @@ -0,0 +1,125 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------ BJT DERIVATIONS -------- +#================================ + +logger.info('Starting BJT DERIVATIONS') + +# ============= +# ---- NPN ---- +# ============= + +logger.info('Starting NPN-BJT DERIVATIONS') + +bjt_exclude = gatpoly.join(pwell_block).join(nsd_drw) + .join(salblock_drw).join(polyres_drw).join(extblock_drw) + .join(res_drw).join(recog_diode).join(recog_esd) + .join(ind_drw).join(ind_pin).join(substrate_drw) + +npn_exclude = nwell_drw.join(psd_drw).join(nbulay_drw).join(bjt_exclude) + +# ---------- General NPN ---------- +npn_mk = trans_drw.and(pwell).and(ptap_holes) +npn_c_exc = emwind_drw.join(emwihv_drw).join(activ_mask) + .join(nsd_block).join(npn_exclude) +npn_b_exc = emwind_drw.join(emwihv_drw).join(npn_exclude) +npn_sub = npn_mk.not(npn_exclude) +npn_dev = activ.join(activ_mask).and(npn_mk) + +# ---------- npn13G2 ---------- +# npn13G2 exclusion layers +npn13G2_e_exc = activ.join(emwihv_drw).join(npn_exclude) +npn13G2_b_exc = npn_b_exc.join(activ_mask) + +# npn13G2 nodes +npn13G2_e_ = emwind_drw.and(activ_mask).and(nsd_block).and(npn_mk).not(npn13G2_e_exc) +# npn13G2 is a fixed device (0.07um X 0.9um) +npn13G2_e_pin = npn13G2_e_.with_bbox_min(0.07.um).with_bbox_max(0.9.um).with_area(0.063.um) +npn13G2_b_pin = nsd_block.and(npn_mk).not(npn13G2_b_exc) +npn13G2_c_pin = activ.and(npn_mk).not_overlapping(npn_c_exc) + +npn13G2_dev = npn_dev.join(nsd_block).extents.covering(npn13G2_e_pin).covering(npn13G2_b_pin).covering(npn13G2_c_pin) +npn13G2_c = npn13G2_dev.sized(-1.nm) +npn13G2_tc = npn13G2_dev.not(npn13G2_c).interacting(npn13G2_c_pin) +npn13G2_b = npn13G2_dev.not(npn13G2_c_pin) +npn13G2_tb = npn13G2_b.not(npn13G2_e_pin).merged +npn13G2_e = npn13G2_e_pin +npn13G2_te = npn13G2_e + +# ---------- npn13G2L ---------- +# npn13G2L exclusion layers +npn13G2l_e_exc = activ_mask.join(nsd_block).join(emwihv_drw).join(npn_exclude) +npn13G2l_b_exc = npn_b_exc.join(activ).join(nsd_block) + +# npn13G2L nodes +npn13G2l_e_ = emwind_drw.and(activ).and(npn_mk).not(npn13G2l_e_exc) +# npn13G2L has fixed width (0.07um), Length could vary from 1:2.5 um +npn13G2l_e_pin = npn13G2l_e_.with_bbox_min(0.07.um).with_bbox_max(1.um, 2.5.um).with_area(0.07.um, 0.175.um) +npn13G2l_b_pin = activ_mask.and(npn_mk).not(npn13G2l_b_exc) +npn13G2l_c_pin = npn13G2_c_pin + +npn13G2l_dev = npn_dev.covering(npn13G2l_e_pin).covering(npn13G2l_b_pin).covering(npn13G2l_c_pin) +npn13G2l_c = npn13G2l_dev.sized(1.nm) +npn13G2l_tc = npn13G2l_c.not(npn13G2l_dev).interacting(npn13G2l_c_pin) +npn13G2l_b = npn13G2l_dev.not(npn13G2l_c_pin) +npn13G2l_tb = npn13G2l_b.not(npn13G2l_e_pin).merged +npn13G2l_e = npn13G2l_e_pin +npn13G2l_te = npn13G2l_e + +# # ---------- npn13G2V ---------- +# npn13G2V exclusion layers +npn13G2v_e_exc = activ_mask.join(nsd_block).join(emwind_drw).join(npn_exclude) + +# npn13G2V nodes +npn13G2v_e_ = emwihv_drw.and(activ).and(npn_mk).not(npn13G2v_e_exc) +# npn13G2L has fixed width (0.12um), Length could vary from 1:5 um +npn13G2v_e_pin = npn13G2v_e_.with_bbox_min(0.12.um).with_bbox_max(1.um, 5.um).with_area(0.12.um, 0.6.um) +npn13G2v_b_pin = npn13G2l_b_pin +npn13G2v_c_pin = npn13G2l_c_pin + +npn13G2v_dev = npn_dev.covering(npn13G2v_e_pin).covering(npn13G2v_b_pin).covering(npn13G2v_c_pin) +npn13G2v_c = npn13G2v_dev.sized(1.nm) +npn13G2v_tc = npn13G2v_c.not(npn13G2v_dev).interacting(npn13G2v_c_pin) +npn13G2v_b = npn13G2v_dev.not(npn13G2v_c_pin) +npn13G2v_tb = npn13G2v_b.not(npn13G2v_e_pin).merged +npn13G2v_e = npn13G2v_e_pin +npn13G2v_te = npn13G2v_e + +# ============= +# ---- PNP ---- +# ============= + +logger.info('Starting PNP-BJT DERIVATIONS') + +pnp_exclude = trans_drw.join(emwind_drw) + .join(emwihv_drw).join(nsd_block).join(bjt_exclude) + +pnp_mk = ptap_holes.not(pnp_exclude) + +# pnp general nodes DERIVATIONS +pnp_e = pactiv.and(pnp_mk).and(nwell_iso) +pnp_b = nactiv.and(pnp_mk).and(nwell_iso) +pnp_c = ptap.interacting(pnp_mk).not(pnp_exclude) + +# pnp_mpa nodes DERIVATIONS +pnp_mpa_e = pnp_e.and(pnp_b.extents).and(pnp_c.extents) +pnp_mpa_b = pnp_b.interacting(pnp_b.extents.interacting(pnp_mpa_e)) +pnp_mpa_c = pnp_c.interacting(pnp_c.extents.interacting(pnp_mpa_e)) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_extraction.lvs new file mode 100644 index 00000000..d372f5b5 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/bjt_extraction.lvs @@ -0,0 +1,81 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------- BJT EXTRACTION -------- +#================================ + +logger.info('Starting BJT EXTRACTION') + +# ============= +# ---- NPN ---- +# ============= + +logger.info('Starting NPN-BJT EXTRACTION') + +logger.info('Extraction of npn13G2 BJT transistor') +extract_devices(bjt4('npn13G2', CustomBJT4), { + 'C' => npn13G2_c, + 'B' => npn13G2_b, + 'E' => npn13G2_e, + 'S' => npn_sub, + 'tC' => npn13G2_tc, + 'tB' => npn13G2_tb, + 'tE' => npn13G2_te, + 'tS' => npn_sub + }) + +logger.info('Extraction of npn13G2L BJT transistor') +extract_devices(bjt4('npn13G2l', CustomBJT4), { + 'C' => npn13G2l_c, + 'B' => npn13G2l_b, + 'E' => npn13G2l_e, + 'S' => npn_sub, + 'tC' => npn13G2l_tc, + 'tB' => npn13G2l_tb, + 'tE' => npn13G2l_te, + 'tS' => npn_sub + }) + +logger.info('Extraction of npn13G2V BJT transistor') +extract_devices(bjt4('npn13G2v', CustomBJT4), { + 'C' => npn13G2v_c, + 'B' => npn13G2v_b, + 'E' => npn13G2v_e, + 'S' => npn_sub, + 'tC' => npn13G2v_tc, + 'tB' => npn13G2v_tb, + 'tE' => npn13G2v_te, + 'tS' => npn_sub + }) + +# ============= +# ---- PNP ---- +# ============= + +logger.info('Starting PNP-BJT EXTRACTION') + +# pnp_mpa BJT +logger.info('Extracting pnpMPA BJT') +extract_devices(bjt3('pnpMPA', CustomBJT3), { 'C' => pnp_mpa_c.extents, + 'B' => pnp_mpa_b.extents, + 'E' => pnp_mpa_e, + 'tC' => pnp_mpa_c, + 'tB' => pnp_mpa_b, + 'tE' => pnp_mpa_e }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_connections.lvs new file mode 100644 index 00000000..645dde29 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_connections.lvs @@ -0,0 +1,39 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================== +# ------ MIMCAP CONNECTIONS ------- +#================================== + +logger.info('Starting LVS CAP CONNECTIONS') + +# === cap_mim === +connect(cmim_btm, metal5_con) +connect(cmim_top, mim_via) +connect(mim_via, topmetal1_con) + +# === rfcmim === +connect(rfmim_btm, metal5_con) +connect(rfmim_top, mim_via) +connect(rfmim_sub, ptap) + +# === svarivap === +connect(varicap_ports, cont_drw) +connect(varicap_ports, text_drw) +connect(varicap_sub, ptap) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_derivations.lvs new file mode 100644 index 00000000..e8a3194e --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_derivations.lvs @@ -0,0 +1,64 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------- CAP DERIVATIONS ------- +#================================ + +logger.info('Starting CAP DERIVATIONS') + +rfmimcap_exc = ind_drw.join(ind_pin) + +# === MIMCAP === +mimcap_exclude = pwell_block.join(rfmimcap_exc) + +mim_top = mim_drw.overlapping(topmetal1_con).and(metal5_con) +mim_btm = metal5_con.and(mim_drw).sized(0.6.um) +mim_via = vmim_drw.join(topvia1_drw).and(mim_drw) +topvia1_n_cap = topvia1_drw.not(mim_via) + +# === cap_cmim === +cmim_top = mim_top.not(mimcap_exclude) +cmim_btm = mim_btm.covering(cmim_top) +cmim_dev = mim_drw.covering(cmim_top).and(cmim_btm) + +# === rfcmim === +rfmim_area = pwell_block.interacting(mim_drw) +rfmim_top = mim_top.and(rfmim_area).not(rfmimcap_exc) +rfmim_btm = mim_btm.and(rfmim_area).covering(rfmim_top) +rfmim_dev = mim_drw.covering(rfmim_top).and(rfmim_btm) +rfmim_sub = ptap.extents.interacting(rfmim_area) +rfmeas_mk = metal5_con.overlapping(rfmim_btm).and(rfmim_area) + +# === svaricap === +cap_exc = nsd_drw.join(trans_drw).join(emwind_drw) + .join(emwihv_drw).join(salblock_drw).join(polyres_drw) + .join(extblock_drw).join(res_drw).join(activ_mask) + .join(recog_diode).join(recog_esd).join(ind_drw) + .join(ind_pin).join(substrate_drw) + +varicap_exc = pwell.join(pwell_block).join(nwell_holes).join(cap_exc) + +varicap_core = ngate_hv_base.and(nwell_iso).not(varicap_exc) +varicap_diff_port = nactiv.interacting(varicap_core).not(varicap_core) + .and(nwell_iso).not(varicap_exc).sized(-1.nm) +varicap_poly_port = gatpoly.interacting(varicap_core) +varicap_ports = varicap_poly_port.join(varicap_diff_port) +varicap_sub = ptap.and(thickgateox_drw) +varicap_dev_mk = thickgateox_drw.covering(varicap_core).interacting(varicap_ports) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_extraction.lvs new file mode 100644 index 00000000..00c8d924 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/cap_extraction.lvs @@ -0,0 +1,55 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================= +# -------- CAP EXTRACTION -------- +#================================= + +logger.info('Starting CAP EXTRACTION') + +# === cap_cmim === +logger.info('Extracting cap_cmim device') +extract_devices(MIMCAPExtractor.new('cap_cmim'), { + 'core' => cmim_top, + 'top_mim' => cmim_top, + 'btm_mim' => cmim_btm, + 'meas_mk' => cmim_dev, + 'dev_mk' => cmim_dev, + }) + +# === rfcmim === +logger.info('Extracting rfcmim device') +extract_devices(MIMCAPExtractor.new('rfcmim'), { + 'core' => rfmim_top, + 'top_mim' => rfmim_top, + 'btm_mim' => rfmim_btm, + 'dev_mk' => rfmim_area.covering(rfmim_dev), + 'meas_mk' => rfmeas_mk, + 'sub_mk' => rfmim_sub, + }) + +# === svaricap === +logger.info('Extracting sg13_hv_svaricap varactor') +extract_devices(GeneralNTerminalExtractor.new('sg13_hv_svaricap', 3), { + 'core' => varicap_core, + 'ports' => varicap_ports, + 'meas_mk' => activ, + 'dev_mk' => varicap_dev_mk, + 'sub_mk' => varicap_sub + }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_classes.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_classes.lvs new file mode 100644 index 00000000..07879748 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_classes.lvs @@ -0,0 +1,55 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +$logger = logger +$unit = dbu + +#================================================ +# --------------- CUSTOM CLASSES ---------------- +#================================================ + +#=============== UTILS =============== +# Method to convert glob pattern to a case-insensitive glob-style pattern +def glob_to_case_insensitive_glob(glob) + wildcards = ['*', '?'] + + pattern = glob.chars.map do |c| + if c =~ /[A-Za-z]/ + "[#{c.upcase}#{c.downcase}]" + elsif wildcards.include?(c) + c # Keep wildcard characters as they are + else + Regexp.escape(c) + end + end.join + + pattern +end + +#=============== CUSTOM READER =================== + +# %include custom_reader.lvs + +#=============== CUSTOM WRITER =================== + +# %include custom_writer.lvs + +#=============== CUSTOM DEVICE =================== + +# %include custom_devices.lvs diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_combiner.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_combiner.lvs new file mode 100644 index 00000000..8ef59f44 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_combiner.lvs @@ -0,0 +1,199 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +$logger = logger +$unit = dbu + +#================================================ +# -------------- CUSTOM COMBINER ---------------- +#================================================ + +# common methods +module DeviceCombinerMethods + private + + # A helper function to check whether two nets are the same + def same_net(a, b, name) + a_net = a.net_for_terminal(name) + b_net = b.net_for_terminal(name) + + # same polarity + same_po = a_net.expanded_name == b_net.expanded_name if a_net && b_net + a_net && b_net && same_po + end + + # A helper function to check whether two device connected in parallel + def supp_parallel(a, b, net1, net2) + a_net1 = a.net_for_terminal(net1) + a_net2 = a.net_for_terminal(net2) + b_net1 = b.net_for_terminal(net1) + b_net2 = b.net_for_terminal(net2) + + return false unless a_net1 && b_net1 && a_net2 && b_net2 + + same_po = a_net1.expanded_name == b_net1.expanded_name && a_net2.expanded_name == b_net2.expanded_name + diff_po = a_net1.expanded_name == b_net2.expanded_name && a_net2.expanded_name == b_net1.expanded_name + + same_po || diff_po + end + + # A helper function to check whether two device connected in series + def supp_series(a, b, net1, net2) + a_net1 = a.net_for_terminal(net1) + a_net2 = a.net_for_terminal(net2) + b_net1 = b.net_for_terminal(net1) + b_net2 = b.net_for_terminal(net2) + + return false unless a_net1 && b_net1 && a_net2 && b_net2 + + series_con_a1 = a_net1.expanded_name == b_net1.expanded_name || a_net1.expanded_name == b_net2.expanded_name + series_con_a2 = a_net2.expanded_name == b_net1.expanded_name || a_net2.expanded_name == b_net2.expanded_name + series_con = series_con_a1 || series_con_a2 + + # Reroute terminal connections based on cluster IDs + if series_con_a1 + if a_net1.expanded_name == b_net1.expanded_name + a.connect_terminal(0, b_net2) + else + a.connect_terminal(0, b_net1) + end + end + + if series_con_a2 + if a_net2.expanded_name == b_net1.expanded_name + a.connect_terminal(1, b_net2) + else + a.connect_terminal(1, b_net1) + end + end + + series_con + end + + # A helper function to check whether two parameters have approximately the same value + def same_parameter(a, b, name) + (a.parameter(name) - b.parameter(name)).abs < 1e-9 + end +end + +class MIMCAPNDeviceCombiner < RBA::GenericDeviceCombiner + include DeviceCombinerMethods + + # Method to check and perform device combination + def combine_devices(a, b) + # Check if both devices have the same net + return false unless same_net(a, b, 'mim_top') && same_net(a, b, 'mim_btm') + + # Check if parameters are the same + return false unless same_parameter(a, b, 'w') && same_parameter(a, b, 'l') + + # Combine by summing up 'm' parameter + a.set_parameter('m', a.parameter('m') + b.parameter('m')) + + # Disconnect the second device and let the system clean it up + b.disconnect_terminal('mim_top') + b.disconnect_terminal('mim_btm') + + # Disconnect mim_sub terminal if present and model name includes 'rfcmim' + b.disconnect_terminal('mim_sub') if b.name.downcase.include?('rfcmim') + + true + end +end + +class DiodeDeviceCombiner < RBA::GenericDeviceCombiner + include DeviceCombinerMethods + + # Method to check and perform device combination + def combine_devices(a, b) + # Check if both devices have the same net + return false unless same_net(a, b, 'A') && same_net(a, b, 'C') + + # Check if parameters are the same + return false unless same_parameter(a, b, 'A') && same_parameter(a, b, 'P') + + # Combine by summing up 'm' parameter + a.set_parameter('m', a.parameter('m') + b.parameter('m')) + + # Disconnect the second device and let the system clean it up + b.disconnect_terminal('A') + b.disconnect_terminal('C') + + true + end +end + +class BJTDeviceCombiner < RBA::GenericDeviceCombiner + include DeviceCombinerMethods + + # Method to check and perform device combination + def combine_devices(a, b) + bjt3_nets = %w[C B E] + bjt4_nets = %w[C B E S] + + # Determine the correct nets based on device type (assuming PNP or NPN) + bjt_nets = a.device_class.name.downcase.include?('pnp') ? bjt3_nets : bjt4_nets + + # Check if terminals have the same net + return false unless bjt_nets.all? { |net| same_net(a, b, net) } + + # Check if parameters are the same + return false unless %w[AE PE].all? { |param| same_parameter(a, b, param) } + + # Combine parameters + a.set_parameter('m', a.parameter('m') + b.parameter('m')) + a.set_parameter('NE', a.parameter('NE') + b.parameter('NE')) + + # Disconnect the second device and let the system clean it up + bjt_nets.each { |term| b.disconnect_terminal(term) } + + true + end +end + +class RESDeviceCombiner < RBA::GenericDeviceCombiner + include DeviceCombinerMethods + + # Method to check and perform device combination + def combine_devices(a, b) + res_nets = [0, 1] # Using id instead of names + + # Check if same parameters + + # Check if terminals series or parallel (With same params) + if supp_parallel(a, b, 0, 1) + return false unless PARALLEL_RES + return false unless %w[w l ps b].all? { |param| same_parameter(a, b, param) } + + a.set_parameter('m', a.parameter('m') + b.parameter('m')) + elsif supp_series(a, b, 0, 1) + return false unless SERIES_RES + return false unless %w[w ps b m].all? { |param| same_parameter(a, b, param) } + + a.set_parameter('l', a.parameter('l') + b.parameter('l')) + else + return false + end + + # Disconnect the second device and let the system clean it up + res_nets.each { |term| b.disconnect_terminal(term) } + + true + end +end diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_devices.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_devices.lvs new file mode 100644 index 00000000..cd812c02 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_devices.lvs @@ -0,0 +1,225 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +$logger = logger +$unit = dbu + +#================================================ +# --------------- CUSTOM DEVICES ---------------- +#================================================ + +# common methods +module DeviceClassMethods + private + + def add_parameters(*names) + names.each { |name| add_parameter(RBA::DeviceParameterDefinition.new(name)) } + end + + def add_terminals(name, num, sub_en) + (1..num).each do |i| + terminal_name = "#{name}_#{i}" + ter = add_terminal(RBA::DeviceTerminalDefinition.new(terminal_name)) + ter.name = terminal_name + end + return unless sub_en == 1 + + terminal_name = "#{name}_sub" + ter = add_terminal(RBA::DeviceTerminalDefinition.new(terminal_name)) + ter.name = terminal_name + end +end + +# res-custom device calss +class DeviceCustomRes < RBA::DeviceClassResistor + include DeviceClassMethods + + def initialize(name, num) + # clear terminals and parameters of resistor class + clear_parameters + clear_terminals + + add_parameters('w', 'l', 'ps', 'b') + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + + if SERIES_RES || PARALLEL_RES + self.combiner = RESDeviceCombiner.new + else + self.combiner =nil + end + + add_terminals(name, num, 0) + end +end + +# BJT-3term device calss +class CustomBJT3 < RBA::DeviceClassBJT3Transistor + + def initialize + super + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + self.combiner = BJTDeviceCombiner.new + + enable_parameter('AE', true) + enable_parameter('PE', true) + enable_parameter('NE', true) + end +end + +# BJT-4term device calss +class CustomBJT4 < RBA::DeviceClassBJT4Transistor + include DeviceClassMethods + + def initialize + super + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + self.combiner = BJTDeviceCombiner.new + + enable_parameter('AE', true) + enable_parameter('PE', true) + enable_parameter('NE', true) + end +end + +# inductor device calss +class DeviceCustomInd < RBA::DeviceClassInductor + include DeviceClassMethods + + def initialize(name, num) + # clear terminals and parameters of class + clear_parameters + clear_terminals + clear_equivalent_terminal_ids + + add_parameters('w', 's', 'd', 'nr_r') + add_terminals(name, num, 1) + + # 5% tolerance for w,s,d: + equal_ind_parameters = RBA::EqualDeviceParameters::new(parameter_id('w'), 0.0, 0.05) + equal_ind_parameters += RBA::EqualDeviceParameters::new(parameter_id('s'), 0.0, 0.05) + equal_ind_parameters += RBA::EqualDeviceParameters::new(parameter_id('d'), 0.0, 0.05) + # applies the compare delegate: + self.equal_parameters = equal_ind_parameters + + self.combiner = nil + self.supports_serial_combination=false + self.supports_parallel_combination=false + end +end + +# Varactor-custom device calss +class DeviceCustomVaractor < RBA::DeviceClassCapacitorWithBulk + include DeviceClassMethods + + def initialize(name, num) + # clear terminals and parameters of resistor class + clear_parameters + clear_terminals + + add_parameters('w', 'l') + add_terminals(name, num, 1) + end +end + +# res-2term device calss +class RES2 < RBA::DeviceClassResistor + def initialize + super + enable_parameter('W', true) + enable_parameter('L', true) + enable_parameter('R', false) + end +end + +# Diode device class +class EnDiode < RBA::DeviceClassDiode + def initialize + + # 1% tolerance for A,P: + equal_diode_parameters = RBA::EqualDeviceParameters::new(parameter_id('A'), 0.0, 0.01) + equal_diode_parameters += RBA::EqualDeviceParameters::new(parameter_id('P'), 0.0, 0.01) + # applies the compare delegate: + self.equal_parameters = equal_diode_parameters + + # combiner + self.combiner = DiodeDeviceCombiner.new + self.supports_serial_combination=true + self.supports_parallel_combination=true + + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + enable_parameter('A', true) + enable_parameter('P', true) + enable_parameter('m', true) + end +end + +# schottky device class +class CustomSchottky < RBA::DeviceClassBJT3Transistor + def initialize + super + clear_parameters + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + + self.combiner = SchottckyDeviceCombiner.new + end +end + +# Taps device class +class CustomTap < RBA::DeviceClassDiode + def initialize + super + clear_parameters + add_parameter(RBA::DeviceParameterDefinition.new('A', 'Area', 0, true)) + add_parameter(RBA::DeviceParameterDefinition.new('P', 'Perimeter', 0, true)) + enable_parameter('A', true) + enable_parameter('P', true) + end +end + +# ESD-3term device class +class Esd3Term < RBA::DeviceClassBJT3Transistor + def initialize + super + clear_parameters + add_parameter(RBA::DeviceParameterDefinition.new('m', 'area', 1, true)) + enable_parameter('m', true) + end +end + +# ESD-2term device class +class Esd2Term < RBA::DeviceClassDiode + def initialize + super + clear_parameters + add_parameter(RBA::DeviceParameterDefinition.new('m', 'area', 1, true)) + enable_parameter('m', true) + end +end + +#=========== CUSTOM COMBINER =========== + +# %include custom_combiner.lvs + +#=========== CUSTOM EXTRACTOR =========== + +# %include custom_extractor.lvs + +#========= CUSTOM MIM-EXTRACTOR ========= + +# %include custom_mim_extractor.lvs diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_extractor.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_extractor.lvs new file mode 100644 index 00000000..55a7a4e8 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_extractor.lvs @@ -0,0 +1,405 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +$logger = logger +$unit = dbu + +#================================================ +# --------------- CUSTOM DEVICES ---------------- +#================================================ + +# === GeneralNTerminalExtractor === +class GeneralNTerminalExtractor < RBA::GenericDeviceExtractor + # Extraction of N terminal devices - General Class + + def initialize(name, num) + # Initialize the extractor with a device name and number of terminals. + # + # Args: + # name (String): The name of the device. + # num (Integer): Number of terminals. + create + self.name = name + @num = num + @name = name + end + + def setup + # Set up layers and register device class for extraction. + define_layers + # Register device class for extraction. + if RES_DEV.any? { |res| name.downcase.start_with?(res) } + @reg_dev = DeviceCustomRes.new(name, @num) + elsif name.downcase.include?('varicap') + @reg_dev = DeviceCustomVaractor.new(name, @num) + elsif name.downcase.start_with?('ind') + @reg_dev = DeviceCustomInd.new(name, @num) + else + raise ArgumentError, "Custom-Class for #{name} device is not supported yet, please recheck" + end + register_device_class(@reg_dev) + end + + def get_connectivity(_layout, layers) + # Establish connectivity between layers. + # + # Args: + # _layout: Layout object (unused). + # layers (Array): Array of layer objects. + # + # Returns: + # Connectivity object representing the connections between layers. + dev = layers[0] + ports = layers[1] + meas_mk = layers[2] + dev_mk = layers[3] + + conn = RBA::Connectivity.new + conn.connect(dev, dev) + conn.connect(dev, dev_mk) + conn.connect(dev, meas_mk) + conn.connect(ports, dev_mk) + conn.connect(meas_mk, dev_mk) + + # Sub connection for some devices + if name.downcase.start_with?('ind') || name.downcase.include?('varicap') + sub_mk = layers[4] + conn.connect(sub_mk, dev_mk) + end + + conn + end + + def extract_devices(layer_geometry) + # Extract devices based on layer geometry. + # + # Args: + # layer_geometry (Array): Array of layer geometries. + dev, ports, meas_mk, dev_mk, sub_mk = layer_geometry + + dev_mk.merged.each do |region| + if ports.size != @num + $logger.info("#{@name} device terminals (#{@num}) don't touch device marker correctly") + $logger.info("No. of ports exist for #{@name} is #{ports.size}, should be #{@num}") + else + device = create_device + set_device_parameters(device, region, dev, ports, meas_mk, dev_mk) + define_and_sort_terminals(device, ports, sub_mk) + end + end + end + + private + + def define_layers + # Define layers for extraction. + define_layer('core', 'core Layer') + define_layer('ports', 'Connect Terminal') + define_layer('meas_mk', 'Measuring parameters marker') + define_layer('dev_mk', 'Device Marker') + # Define sub layer for some devices + if name.downcase.start_with?('ind') || name.downcase.include?('varicap') + define_layer('sub_mk', 'Substrate Marker') + end + end + + def set_device_parameters(device, region, dev, ports, meas_mk, dev_mk) + # Set device parameters based on device type. + # + # Args: + # device: Device object to set parameters for. + # region: Region representing the measured region. + # dev: Device layer object. + # ports: ports layer object. + # meas_mk: Measuring marker layer object. + # dev_mk: main marker layer object. + # + # Returns: + # None + + if RES_DEV.any? { |res| name.downcase.start_with?(res) } + width, length, poly_sp, bends = calc_res_params(dev, ports, meas_mk) + device.set_parameter('w', width * $unit) + device.set_parameter('l', length * $unit) + device.set_parameter('ps', poly_sp * $unit) + device.set_parameter('b', bends) + + elsif name.downcase.include?('varicap') + width, length = calc_varicap_params(dev, ports, meas_mk, dev_mk) + device.set_parameter('w', width * $unit) + device.set_parameter('l', length * $unit) + + elsif name.downcase.start_with?('ind') + width, space, diameter, no_turns = calc_ind_params(dev, ports, meas_mk, dev_mk, region) + device.set_parameter('w', width * $unit) + device.set_parameter('s', space * $unit) + device.set_parameter('d', diameter * $unit) + device.set_parameter('nr_r', no_turns) + end + end + + def calc_res_params(dev, ports, meas_mk) + # Width + width_edges = dev.edges.and(ports.edges) + width = get_uniq_length(width_edges) + + # Length + length_edges = dev.edges.interacting(width_edges).not(width_edges) + length, _ = get_min_max_length(length_edges) + + # Bends + corners = meas_mk.interacting(dev).corners.not_interacting(ports).count + bends = corners / 4 + + # poly_space between bends + if bends.positive? + poly_sp_polygon = meas_mk.interacting(dev) + poly_sp = get_notch_min(poly_sp_polygon, 10 * length) + length = length + width + end + + # Default values + width ||= 0 + length ||= 0 + poly_sp ||= 0 + bends ||= 0 + + [width, length, poly_sp, bends] + end + + def calc_varicap_params(dev, _ports, meas_mk, _dev_mk) + # Width & Length + width_edges = dev.edges.not_interacting(meas_mk.edges) + length, width = get_min_max_length(width_edges) + + # Default values + width ||= 0 + length ||= 0 + + [width, length] + end + + def calc_ind_params(dev, ports, meas_mk, dev_mk, region) + # Get upper limit for width, space + _, max_mk_len = get_min_max_length(dev_mk.edges) + + # Width + meas_sel = meas_mk.merged & region + width = get_width_val(meas_sel, max_mk_len) + + # space + space = get_space_val(meas_sel, max_mk_len) + + # Turns + # Calc steps used for no. of turns: + # step1: Get count of inductor metal (catch if we have more 1) + # Step2: For more than 1 turns, get number of holes + # Step3: Turns = 1 + (holes - 1)/2 + no_turns_init = meas_mk.merged.count + no_turns = no_turns_init + + ## Old implementation + # if no_turns_init == 1 + # no_turns = no_turns_init + # else + # no_turns_pre1 = dev.merged.holes.count + # no_turns_pre2 = (no_turns_pre1 - 1) / 2 + # no_turns = 1 + no_turns_pre2.ceil + # end + + # Diameter + # Calc steps used for diameter: + # step1: Get extent of the inductor core + # step2: Exclude edges that touch inductor pins + # step3: Get length of the remaining edge (Outer diameter) + # step4: Get internal diameter --> din = dout - ((turns -1) * 2s) - (turns * w) + diam_extents = dev.extents.edges + diam_edge_exc = diam_extents.interacting(ports) + diam_edge = diam_extents.not_interacting(diam_edge_exc) + diameter = diam_edge.length + diameter = diameter - (2 * (no_turns - 1) * space) - (2 * no_turns * width) + diameter = diameter.negative? ? 0 : diameter + + # Default values + width ||= 0 + space ||= 0 + diameter ||= 0 + no_turns ||= 1 + + [width, space, diameter, no_turns] + end + + def define_and_sort_terminals(device, ports, sub_mk) + # Define and sort terminals based on location. + # + # Args: + # device: Device object to define terminals for. + # ports: Contact layer object containing terminals. + # sub_mk: substrate marker layer object. + # + # Returns: + # None + + # If none of the substrings match, sorted_ports remains the result of sort_polygons(ports) + substrings = %w[varicap] + + # Initialize sorted_ports with a default value + sorted_ports = nil + + # Iterate over each substring + substrings.each do |substring| + if name.include?(substring) + sorted_ports = ports + break # Exit loop if a match is found + end + end + + # If none of the substrings match, sorted_ports remains the result of sort_polygons(ports) + sorted_ports ||= sort_polygons(ports) + + # Define sub if exist (should be defined before other terminals) + if name.downcase.start_with?('ind') || name.downcase.include?('varicap') + if sub_mk.is_empty? + $logger.info("Sub terminal for #{@name} device doesn't exist, please recheck") + return nil + else + define_terminal(device, @reg_dev.terminal_id("#{name}_sub"), 4, sub_mk[0]) + end + end + + # Defination main terminals + (1..@num).each do |i| + define_terminal(device, @reg_dev.terminal_id("#{name}_#{i}"), 1, sorted_ports[i - 1]) + end + end + + def sort_polygons(polygons) + # Sort polygons points. + # + # Args: + # polygons: Polygons to sort. + # + # Returns: + # Sorted polygons. + # + # Note: + # This function sorts the points of the input polygons to be ordered as expected. + # It takes an array of polygons and returns the sorted array. + # The sorting is based on the x-coordinate of the first point of each polygon. + con_polygons = [] + + polygons.merged.each do |ports_pl| + con_edges = [] + ports_pl.each_edge do |con_ed| + con_edges.append([con_ed.x1, con_ed.y1]) + con_edges.append([con_ed.x2, con_ed.y2]) + end + con_polygons.append(con_edges.uniq) + end + sorted_ports_polygons = con_polygons.sort_by(&:first) + sorted_ports = [] + sorted_ports_polygons.each do |sorted_pl| + ports_pl = RBA::DPolygon.new([RBA::DPoint.new(sorted_pl[0][0], sorted_pl[0][1]), + RBA::DPoint.new(sorted_pl[1][0], sorted_pl[1][1]), + RBA::DPoint.new(sorted_pl[2][0], sorted_pl[2][1]), + RBA::DPoint.new(sorted_pl[3][0], sorted_pl[3][1])]) + sorted_ports.append(ports_pl) + end + + sorted_ports + end + + def get_uniq_length(sel_edges) + # Extract uniqe length value for some selected edges + lengths = [] + sel_edges.each do |edge| + lengths << edge.length + end + lengths.uniq! + lengths.size == 1 ? lengths[0] : 0.0 + end + + def get_sep_val(sel_edges, sep_edges, sep_val) + # Extract distance between edges for separation check + proj = RBA::Metrics::Projection + sep_paris = sel_edges.separation_check(sep_edges, sep_val, proj) + sep_values = [] + sep_paris.each do |edge| + sep_values << edge.distance + end + sep_values.min + end + + def get_space_val(sel_polygon, sep_val) + # Extract distance between edges for space check + proj = RBA::Metrics::Projection + space_paris = sel_polygon.space_check(sep_val, proj) + space_values = [] + space_paris.each do |edge| + space_values << edge.distance + end + space_values.min + end + + def get_width_val(sel_polygon, width_val) + # intra-polygon spacing check + proj = RBA::Metrics::Projection + width_paris = sel_polygon.width_check(width_val, metrics: proj, min_projection: 10) + width_values = [] + width_paris.each do |edge| + width_values << edge.distance + end + # Group the array elements by their occurrences + width_values = width_values.reject(&:zero?) + + width_values.min + end + + def get_notch_min(sel_polygon, sep_val) + # intra-polygon spacing check + proj = RBA::Metrics::Projection + space_paris = sel_polygon.notch_check(sep_val, proj) + space_values = [] + space_paris.each do |edge| + space_values << edge.distance + end + space_values.min + end + + def get_notch_max(sel_polygon, sep_val) + # intra-polygon spacing check + proj = RBA::Metrics::Projection + space_paris = sel_polygon.notch_check(sep_val, proj) + space_values = [] + space_paris.each do |edge| + space_values << edge.distance + end + space_values.max + end + + def get_min_max_length(sel_edges) + # Extract max length value for some selected edges + lengths = [] + sel_edges.each do |edge| + lengths << edge.length + end + lengths.minmax + end +end + diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_mim_extractor.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_mim_extractor.lvs new file mode 100644 index 00000000..60d39712 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_mim_extractor.lvs @@ -0,0 +1,245 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +$logger = logger +$unit = dbu + +#================================================ +# --------------- CUSTOM DEVICES ---------------- +#================================================ + +# MIM-custom device calss +class DeviceCustomMIM < RBA::DeviceClassCapacitor + def initialize(name) + # clear terminals and parameters of resistor class + clear_parameters + clear_terminals + clear_equivalent_terminal_ids + + # Adding params + add_parameter(RBA::DeviceParameterDefinition.new('w', 'width', 0, false)) + add_parameter(RBA::DeviceParameterDefinition.new('l', 'length', 0, false)) + add_parameter(RBA::DeviceParameterDefinition.new('A', 'area', 0, true)) + add_parameter(RBA::DeviceParameterDefinition.new('P', 'perimeter', 0, true)) + add_parameter(RBA::DeviceParameterDefinition.new('m', 'multiplier', 1, true)) + + # Adding terminals + ter1 = add_terminal(RBA::DeviceTerminalDefinition.new("mim_top")) + ter2 = add_terminal(RBA::DeviceTerminalDefinition.new("mim_btm")) + ter1.name = "mim_top" + ter2.name = "mim_btm" + + # Adding extra param & terminal for rfcmim + return unless name.downcase.include?('rfcmim') + + add_parameter(RBA::DeviceParameterDefinition.new('wfeed', 'feed width', 0, true)) + sub_ter = add_terminal(RBA::DeviceTerminalDefinition.new("mim_sub")) + sub_ter.name = "mim_sub" + end +end + +# === MIMCAPExtractor === +class MIMCAPExtractor < RBA::GenericDeviceExtractor + # Extraction of N terminal devices - General Class + + def initialize(name) + # Initialize the extractor with a device name and number of terminals. + # + # Args: + # name (String): The name of the device. + + self.name = name + @name = name + end + + def setup + # Set up layers and register device class for extraction. + define_layers + # Register device class for extraction. + @reg_dev = DeviceCustomMIM.new(name) + + # Disable combination for rfcmim + if name.downcase.include?('rfcmim') + @reg_dev.combiner = nil + else + @reg_dev.combiner = MIMCAPNDeviceCombiner.new + end + register_device_class(@reg_dev) + end + + def get_connectivity(_layout, layers) + # Establish connectivity between layers. + # + # Args: + # _layout: Layout object (unused). + # layers (Array): Array of layer objects. + # + # Returns: + # Connectivity object representing the connections between layers. + dev = layers[0] + top_mim = layers[1] + btm_mim = layers[2] + dev_mk = layers[3] + meas_mk = layers[4] + + conn = RBA::Connectivity.new + conn.connect(dev, dev) + conn.connect(dev, dev_mk) + conn.connect(dev, meas_mk) + conn.connect(top_mim, dev_mk) + conn.connect(btm_mim, dev_mk) + + # Sub connection for rfcmim + if name.downcase.include?('rfcmim') + sub_mk = layers[5] + conn.connect(sub_mk, dev_mk) + end + + conn + end + + def extract_devices(layer_geometry) + # Extract devices based on layer geometry. + # + # Args: + # layer_geometry (Array): Array of layer geometries. + dev, top_mim, btm_mim, dev_mk, meas_mk, sub_mk = layer_geometry + + dev_mk.merged.each do |_region| + if top_mim.size != 1 + $logger.info("No. of ports exist for #{@name} topmetal is #{top_mim.size}, should be 1") + elsif btm_mim.size != 1 + $logger.info("No. of ports exist for #{@name} btmmetal is #{btm_mim.size}, should be 1") + else + device = create_device + set_device_parameters(device, dev, dev_mk, meas_mk) + define_terminals(device, top_mim, btm_mim, sub_mk) + end + end + end + + private + + def define_layers + # Define layers for extraction. + define_layer('core', 'core Layer') + define_layer('top_mim', 'Connect Terminal for top mim') + define_layer('btm_mim', 'Connect Terminal for btm mim') + define_layer('dev_mk', 'Device Marker') + define_layer('meas_mk', 'Measuring parameters marker') + + # Define sub layer for some devices + return unless name.downcase.include?('rfcmim') + + define_layer('sub_mk', 'Substrate layer') + end + + def set_device_parameters(device, dev, dev_mk, meas_mk) + # Set device parameters based on device type. + # + # Args: + # device: Device object to set parameters for. + # dev: Device layer object. + # dev_mk: device marker layer object. + # meas_mk: meas marker layer object. + # + # Returns: + # None + + width, length, wfeed = calc_cmim_params(dev, dev_mk, meas_mk) + + if name.downcase.include?('rfcmim') + device.set_parameter('l', width * $unit) + device.set_parameter('w', length * $unit) + device.set_parameter('wfeed', wfeed * $unit) + else + device.set_parameter('w', width * $unit) + device.set_parameter('l', length * $unit) + end + device.set_parameter('A', width * length * $unit * $unit) + device.set_parameter('P', (width + length) * 2 * $unit) + + end + + def calc_cmim_params(dev, dev_mk, meas_mk) + # Width & Length + dev_edges = dev.edges + width_edges = dev_edges.with_angle(0, false) + len_edges = dev_edges.not(width_edges) + width = get_uniq_length(width_edges) + length = get_uniq_length(len_edges) + + # Wfeed + wfeed_edges = meas_mk.edges.and(dev_mk.edges) + wfeed = get_uniq_length(wfeed_edges) + + # Default values + width ||= 0 + length ||= 0 + wfeed ||= 0 + + [width, length, wfeed] + end + + def define_terminals(device, top_mim, btm_mim, sub_mk) + # Define terminals based on location. + # + # Args: + # device: Device object to define terminals for. + # top_mim: Contact layer object containing mim top metal. + # top_mim: Contact layer object containing mim btm metal. + # sub_mk: substrate marker layer object. + # + # Returns: + # None + + # Define sub if exist (should be defined before other terminals) + if name.downcase.include?('rfcmim') + if sub_mk.is_empty? + $logger.info("Sub terminal for #{@name} device doesn't exist, please recheck") + return nil + else + define_terminal(device, @reg_dev.terminal_id("mim_sub"), 5, sub_mk[0]) + end + end + + # Defination main terminals + define_terminal(device, @reg_dev.terminal_id("mim_top"), 1, top_mim[0]) + define_terminal(device, @reg_dev.terminal_id("mim_btm"), 2, btm_mim[0]) + end + + def get_min_max_length(sel_edges) + # Extract max length value for some selected edges + lengths = [] + sel_edges.each do |edge| + lengths << edge.length + end + lengths.minmax + end + + def get_uniq_length(sel_edges) + # Extract uniqe length value for some selected edges + lengths = [] + sel_edges.each do |edge| + lengths << edge.length + end + lengths.uniq! + lengths.size == 1 ? lengths[0] : 0.0 + end +end diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_reader.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_reader.lvs new file mode 100644 index 00000000..0b517fba --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_reader.lvs @@ -0,0 +1,346 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=========== CUSTOM READER =========== + +# Custom reader for subcircuit models +class CustomReader < RBA::NetlistSpiceReaderDelegate + # Cleanup sch for R, C elements + def clean_sch(line, element) + # Extracting parameters with values + valid_params = line.scan(/\b\w+\s*=\s*\S+\b/) + + case element + when 'R' + # For 2 term res [ ] + num_terms = 3 + when 'C' + # Determine number of terms based on component type + num_terms = + if line.downcase.include?('varicap') + 5 + elsif line.downcase.include?('rfcmim') + 4 + else + 3 + end + end + line_no_param = line.split(' ').take(num_terms).join(' ') + "#{line_no_param.strip} #{valid_params.join(' ')}" + end + + # Override parse_element method to handle exceptions gracefully + def parse_element(line, element) + # clean line + line = line.delete('[]$\\/') + + # Prep sch for R, C + line = clean_sch(line, element) if %w[R C].include?(element) + + super + rescue StandardError + case element + when 'C' + if line.downcase.include?('varicap') + super(line.to_s, 'M') + elsif line.downcase.include?('rfcmim') + super(line.to_s, 'Q') + else + super("#{line} C=1e-18", element) + end + when 'R' + super("#{line} R=0", element) + when 'D' + if line.downcase.include?('diodev') || line.downcase.include?('schottky') + super(line.to_s, 'Q') + else + super(line.to_s, element) + end + when 'L' + if line.downcase.include?('inductor3') + super("#{line} L=0", 'M') + else + super("#{line} L=0", element) + end + else + super + end + end + + # Override the element method to handle different types of elements + def element(circuit, ele, name, model, value, nets, params) + if CUSTOM_READER.include?(ele) + process_device(ele, circuit, name, model, nets, params) + else + super + end + true + end + + private + + # Process device element + def process_device(ele, circuit, name, model, nets, params) + cls = circuit.netlist.device_class_by_name(model) + cls ||= create_device_class(ele, circuit, model, nets.size) + + device = circuit.create_device(cls, name) + connect_terminals(ele, device, model, nets) + map_params(ele, device, model, params) + end + + # Create or retrieve the device class based on the element type, model name, and number of nets. + # + # @param ele [String] The type of element (C, R, Q, L). + # @param circuit [Circuit] The circuit object to which the device class will be added. + # @param model [String] The model name of the device class. + # @param num_nets [Integer] The number of nets the device class should have. + # @return [RBA::DeviceClass] The created or retrieved device class. + def create_device_class(ele, circuit, model, num_nets) + cls = case ele + when 'M' then RBA::DeviceClassMOS4Transistor.new + when 'C' then create_capacitor(model, num_nets) + when 'R' then create_resistor(model, num_nets) + when 'Q' then create_bjt(model, num_nets) + when 'L' then DeviceCustomInd.new(model, num_nets - 1) + when 'D' then create_diode(model) + else + return super + end + + cls.name = model + circuit.netlist.add(cls) + cls + end + + # Create a capacitor device class. + def create_capacitor(model, num_nets) + if model.downcase.include?('varicap') + raise ArgumentError, 'Varicap should have 4 nodes, please recheck' unless num_nets == 4 + + DeviceCustomVaractor.new(model, num_nets - 1) + else + raise ArgumentError, 'Capacitor should have 2 or 3 nodes, please recheck' unless [2, 3].include?(num_nets) + + DeviceCustomMIM.new(model) + + end + end + + # Create a diode device class. + def create_diode(model) + if model.downcase.include?('diodev') || model.downcase.include?('schottky') + Esd3Term.new + elsif model.downcase.include?('nmoscl') + Esd2Term.new + else + EnDiode.new + end + end + + # Create a resistor device class. + def create_resistor(model, num_nets) + if RES_DEV.any? { |res| model.downcase.start_with?(res) } + DeviceCustomRes.new(model, num_nets) + elsif num_nets == 2 && model.downcase.include?('tap') + RBA::DeviceClassDiode.new + elsif num_nets == 2 + RBA::DeviceClassResistor.new + elsif num_nets == 3 + RBA::DeviceClassResistorWithBulk.new + else + raise ArgumentError, 'Resistor should have two or three nodes, please recheck' + end + end + + # Create a bjt device class. + def create_bjt(model, _num_nets) + if model.downcase.include?('pnp') + CustomBJT3.new + else + CustomBJT4.new + end + end + + # Connect device terminals based on element type, device, model, and nets. + # + # @param ele [String] The type of element (C, R, Q). + # @param device [RBA::Device] The device object to which terminals will be connected. + # @param model [String] The model name of the device. + # @param nets [Array] Array of net names to which terminals will be connected. + def connect_terminals(ele, device, model, nets) + term_list = terminal_list_for_element(ele, model, nets) + + term_list.each_with_index do |t, index| + device.connect_terminal(t, nets[index]) + end + end + + # Determine terminal list based on element type, model, and nets. + def terminal_list_for_element(ele, model, nets) + case ele + when 'M' + %w[D G S B] + when 'Q' + model.downcase.include?('pnp') ? %w[C B E] : %w[C B E S] + when 'C' + model.downcase.include?('varicap') ? gen_term_with_sub(model, nets.size) : gen_mim_terms(model) + when 'R' + if RES_DEV.any? { |res| model.downcase.start_with?(res) } + gen_term_names(model, nets.size) + elsif model.downcase.include?('tap') + %w[C A] + else + nets.size == 3 ? %w[A B W] : %w[A B] + end + when 'D' + if model.downcase.include?('diodevdd') + %w[B E C] + elsif model.downcase.include?('diodevss') + %w[C E B] + elsif model.downcase.include?('schottky') + %w[E B C] + elsif model.downcase.include?('nmoscl') + %w[C A] + else + %w[A C] + end + when 'L' + gen_term_with_sub(model, nets.size) + else + gen_term_names(model, nets.size) + end + end + + # Generate terminal names based on model and the number of nets. + def gen_term_names(model, size) + (0...size).map { |i| "#{model}_#{i + 1}" } + end + + # Generate terminal names based on model and the number of nets. + def gen_mim_terms(model) + terms = %w[mim_top mim_btm] + + return terms unless model.downcase.include?('rfcmim') + + terms << 'mim_sub' # Add sub terminal + terms + end + + # Generate terminal names based on model and the number of nets. + def gen_term_with_sub(model, size) + terms = (0...size - 1).map { |i| "#{model}_#{i + 1}" } + terms << "#{model}_sub" # Add sub terminal + end + + # Map parameters based on the model type. + # + # @param ele [String] The type of element (M, C, R, Q, L, D). + # @param device [RBA::Device] The device object to which parameters will be mapped. + # @param model [String] The model name of the device. + # @param params [Hash] Hash containing parameter values. + def map_params(ele, device, model, params) + case ele + when 'M' + map_mos_params(device, params) + when 'Q' + map_bjt_params(device, model, params) + when 'C' + map_capacitor_params(device, model, params) + when 'R' + map_resistor_params(device, model, params) + when 'D' + map_diode_params(device, model, params) + when 'L' + map_inductor_params(device, params) + else + raise ArgumentError, "#{ele} device with model #{model} is not supported, please recheck" + end + end + + # Map parameters for mos devices. + def map_mos_params(device, params) + device.set_parameter('W', (params['W'] || 0.0) * (params['M'] || 1.0) * 1e6) + device.set_parameter('L', (params['L'] || 0.0) * 1e6) + end + + # Map parameters for a BJT device. + def map_bjt_params(device, model, params) + if model.downcase.include?('pnp') + device.set_parameter('AE', (params['A'] || ((params['W'] || 0.0) * (params['L'] || 0.0))) * 1e12) + device.set_parameter('PE', (params['P'] || (((params['W'] || 0.0) + (params['L'] || 0.0)) * 2)) * 1e6) + else + device.set_parameter('AE', (params['AE'] || ((params['WE'] || 0.0) * (params['LE'] || 0.0))) * 1e12) + device.set_parameter('PE', (params['PE'] || (((params['WE'] || 0.0) + (params['LE'] || 0.0)) * 2)) * 1e6) + end + device.set_parameter('NE', params['M'] || params['NE'] || 1.0) + device.set_parameter('m', params['M'] || params['NE'] || 1.0) + end + + # Map parameters for a diode device. + def map_diode_params(device, model, params) + unless model.downcase.include?('diodev') || model.downcase.include?('schottky') || model.downcase.include?('nmoscl') + device.set_parameter('A', (params['A'] || ((params['W'] || 0.0) * (params['L'] || 0.0))) * 1e12) + device.set_parameter('P', (params['P'] || (((params['W'] || 0.0) + (params['L'] || 0.0)) * 2)) * 1e6) + end + device.set_parameter('m', params['M'] || 1.0) + end + + # Map parameters for a capacitor device. + def map_capacitor_params(device, model, params) + device.set_parameter('w', (params['W'] || 0.0) * 1e6) + device.set_parameter('l', (params['L'] || 0.0) * 1e6) + device.set_parameter('m', params['M'] || params['MF'] || 1.0) if model.downcase.include?('cap_cmim') + + if model.downcase.include?('mim') + device.set_parameter('A', (params['A'] || ((params['W'] || 0.0) * (params['L'] || 0.0))) * 1e12) + device.set_parameter('P', (params['P'] || (((params['W'] || 0.0) + (params['L'] || 0.0)) * 2)) * 1e6) + end + return unless model.downcase.include?('rfcmim') + + device.set_parameter('wfeed', (params['WFEED'] || 0.0) * 1e6) + end + + # Map parameters for a resistor device. + def map_resistor_params(device, model, params) + if model.downcase.include?('tap') + device.set_parameter('A', (params['A'] || ((params['W'] || 0.0) * (params['L'] || 0.0))) * 1e12) + device.set_parameter('P', + (params['P'] || params['PERIM'] || (((params['W'] || 0.0) + (params['L'] || 0.0)) * 2)) * 1e6) + elsif RES_DEV.any? { |res| model.downcase.start_with?(res) } + device.set_parameter('w', (params['W'] || 0.0) * 1e6) + device.set_parameter('l', (params['L'] || 0.0) * 1e6) + device.set_parameter('ps', (params['PS'] || 0.0) * 1e6) + device.set_parameter('b', params['B'] || 0.0) + device.set_parameter('m', params['M'] || 1.0) + else + device.set_parameter('W', (params['W'] || params['WIDTH'] || 0.0) * (params['M'] || 1.0) * 1e6) + device.set_parameter('L', (params['L'] || params['LENGTH'] || 0.0) * (params['S'] || 1.0) * 1e6) + device.set_parameter('R', (params['R'] || 0.0) * (params['S'] || 1.0) / (params['M'] || 1.0)) + end + end + + # Map parameters for an inductor device. + def map_inductor_params(device, params) + device.set_parameter('w', (params['W'] || 0.0) * 1e6) + device.set_parameter('s', (params['S'] || 0.0) * 1e6) + device.set_parameter('d', (params['D'] || 0.0) * 1e6) + device.set_parameter('nr_r', params['NR_R'] || 0.0) + end +end diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_writer.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_writer.lvs new file mode 100644 index 00000000..09c33fe6 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/custom_writer.lvs @@ -0,0 +1,66 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=============== CUSTOM WRITER =================== + +# %include globals.lvs + +# Custom writer for SPICE netlists +class CustomWriter < RBA::NetlistSpiceWriterDelegate + # Write device to SPICE format + # @param device [RBA::Device] The device to be written + def write_device(device) + device_class = device.device_class + str = generate_device_prefix(device, device_class) + str += generate_device_terminals(device, device_class) + str += "#{device_class.name} " + str += generate_default_parameters(device, device_class) + emit_line(str) + end + + private + + # Generate device prefix using the global prefix map + def generate_device_prefix(device, device_class) + prefix = PREFIX_MAP[device_class.name] || device.id.to_s + "#{prefix}#{device.expanded_name} " + end + + # Generate device terminals + def generate_device_terminals(device, device_class) + terminals = device_class.terminal_definitions.map do |td| + net_to_string(device.net_for_terminal(td.id)) + end + "#{terminals.join(' ')} " + end + + # Generate parameters with given keys + def generate_parameters(device, *keys) + parameters = keys.map { |key| "#{key}=#{device.parameter(key)}" } + parameters.join(' ') + end + + # Generate default parameters for the device + def generate_default_parameters(device, device_class) + parameters = device_class.parameter_definitions.map do |pd| + format('%s=%.12g', name: pd.name, value: device.parameter(pd.id)) + end + parameters.join(' ') + end +end diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/devices_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/devices_connections.lvs new file mode 100644 index 00000000..e95b932b --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/devices_connections.lvs @@ -0,0 +1,84 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================================ +#------------ DEVICES CONNECTIVITY -------------- +#================================================ + +logger.info('Starting SG13G2 LVS connectivity setup') + +#================================ +# ----- GENERAL CONNECTIONS ----- +#================================ + +# %include general_connections.lvs + +#================================ +# ----- MOSFET CONNECTIONS ------ +#================================ + +# %include mos_connections.lvs + +#================================ +# ---- RF-MOSFET CONNECTIONS ---- +#================================ + +# %include rfmos_connections.lvs + +#================================ +# ------ BJT CONNECTIONS -------- +#================================ + +# %include bjt_connections.lvs + +#================================ +# ----- DIODE CONNECTIONS ------- +#================================ + +# %include diode_connections.lvs + +#================================ +# ------- RES CONNECTIONS ------- +#================================ + +# %include res_connections.lvs + +#================================== +# -------- CAP CONNECTIONS -------- +#================================== + +# %include cap_connections.lvs + +#================================ +# ------- ESD CONNECTIONS ------- +#================================ + +# %include esd_connections.lvs + +#================================= +# ----- Inductor CONNECTIONS ----- +#================================= + +# %include ind_connections.lvs + +#================================ +# ------- Taps CONNECTIONS ------ +#================================ + +# %include tap_connections.lvs diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_connections.lvs new file mode 100644 index 00000000..ad39928f --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_connections.lvs @@ -0,0 +1,39 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- DIODE CONNECTIONS ------- +#================================ + +logger.info('Starting LVS DIODE CONNECTIONS') + +# dantenna diode +connect(dantenna_n, cont_drw) +connect(dantenna_p, pwell) + +# dantenna diode +connect(dpantenna_n, nwell_drw) +connect(dpantenna_p, cont_drw) + +# dantenna diode +connect(schottcky_p_1x1, schottcky_p) +connect(schottcky_p, metal1_con) +connect(schottcky_n, schottcky_n_port) +connect(schottcky_n_port, schottcky_n_con) +connect(schottcky_n_con, metal1_con) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_derivations.lvs new file mode 100644 index 00000000..3e35fde3 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_derivations.lvs @@ -0,0 +1,64 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- DIODE DERIVATIONS ------- +#================================ + +logger.info('Starting DIODE DERIVATIONS') + +diode_exclude = gatpoly.join(nsd_drw).join(trans_drw) + .join(emwind_drw).join(emwihv_drw).join(polyres_drw) + .join(extblock_drw).join(res_drw).join(activ_mask) + .join(recog_esd).join(ind_drw).join(ind_pin) + .join(substrate_drw) + +antenna_d_exc = pwell_block.join(salblock_drw) + .join(nsd_block).join(diode_exclude) + +antenna_d_mk = recog_diode.not(antenna_d_exc) + +# ==== dantenna diode ==== +dantenna_n = activ.and(antenna_d_mk).not(psd_drw).not(nwell_drw) +dantenna_p = pwell.and(antenna_d_mk).covering(dantenna_n) + +# ==== dpantenna diode ==== +dpantenna_p = pactiv.and(antenna_d_mk) +dpantenna_n = nwell_drw.covering(dpantenna_p) + +# ==== schottky_nbl1 diode ==== +schottky_mk = recog_diode.and(thickgateox_drw).not(diode_exclude) + .and(pwell_block).and(ptap_holes).and(nbulay_drw) + .and(salblock_drw).and(nsd_block).and(nwell_holes) + .not(psd_drw).not(pwell).not(diode_exclude) + +schottcky_p_ = cont_drw.and(activ).and(metal1_con) + .and(schottky_mk) + +# schottky_nbl1 is a fixed device (0.3um X 1.0 um) +schottcky_p = schottcky_p_.with_bbox_min(0.3.um).with_bbox_max(1.0.um) +# Using box with area 1x1 to be used as a reference to (m) +schottcky_p_1x1 = schottcky_p.middle(as_boxes).sized(0.499.um) + +schottcky_n = nsd_block.and(activ).covering(schottcky_p) + +# define port for schottcky +schottcky_n_port = activ.interacting(nwell_iso).interacting(schottcky_n).not(schottcky_n.sized(-1.nm)) +schottcky_n_con = cont_drw.and(schottcky_n_port).not_interacting(schottcky_p) +schottcky_sub = ptap.extents.covering(schottcky_p).covering(schottcky_n) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_extraction.lvs new file mode 100644 index 00000000..13495ccd --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/diode_extraction.lvs @@ -0,0 +1,41 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------ DIODE EXTRACTION ------- +#================================ + +logger.info('Starting DIODE EXTRACTION') + +# dantenna diode +logger.info('Extracting dantenna diode') +extract_devices(diode('dantenna', EnDiode), { 'N' => dantenna_n, 'P' => dantenna_p }) + +# dpantenna diode +logger.info('Extracting dpantenna diode') +extract_devices(diode('dpantenna', EnDiode), { 'N' => dpantenna_n, 'P' => dpantenna_p }) + +# schottky_nbl1 diode +logger.info('Extracting schottky_nbl1 diode') +extract_devices(bjt3('schottky_nbl1', Esd3Term), { 'C' => schottcky_sub, + 'B' => schottcky_n_port.extents, + 'E' => schottcky_p_1x1, + 'tC' => ptap, + 'tB' => schottcky_n_port, + 'tE' => schottcky_p_1x1 }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_connections.lvs new file mode 100644 index 00000000..7c37e0a5 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_connections.lvs @@ -0,0 +1,108 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- RESISTOR CONNECTIONS ----- +#================================ + +logger.info('Starting ESD CONNECTIONS') + +#====================== +# ----- diode-ESD ----- +#====================== + +# diodevdd_2kv +connect(diodevdd_2kv_e_1x1, diodevdd_2kv_e) +connect(diodevdd_2kv_e, cont_drw) +connect(diodevdd_2kv_b, diodevdd_2kv_tb) +connect(diodevdd_2kv_tb, cont_drw) +connect(diodevdd_2kv_c, cont_drw) + +# diodevdd_4kv +connect(diodevdd_4kv_e_1x1, diodevdd_4kv_e) +connect(diodevdd_4kv_e, diodevdd_4kv_te) +connect(diodevdd_4kv_te, cont_drw) +connect(diodevdd_4kv_b, diodevdd_4kv_tb) +connect(diodevdd_4kv_tb, cont_drw) +connect(diodevdd_4kv_c, cont_drw) + +# diodevss_2kv +connect(diodevss_2kv_e_1x1, diodevss_2kv_e) +connect(diodevss_2kv_e, cont_drw) +connect(diodevss_2kv_b, diodevss_2kv_tb) +connect(diodevss_2kv_tb, cont_drw) +connect(diodevss_2kv_c, cont_drw) + +# diodevss_4kv +connect(diodevss_4kv_e_1x1, diodevss_4kv_e) +connect(diodevss_4kv_e, diodevss_4kv_te) +connect(diodevss_4kv_te, cont_drw) +connect(diodevss_4kv_b, diodevss_4kv_tb) +connect(diodevss_4kv_tb, cont_drw) +connect(diodevss_4kv_c, cont_drw) + +#======================= +# ----- idiode-ESD ----- +#======================= + +# idiodevdd_2kv +connect(idiodevdd_2kv_e_1x1, idiodevdd_2kv_e) +connect(idiodevdd_2kv_e, cont_drw) +connect(idiodevdd_2kv_b, idiodevdd_2kv_tb) +connect(idiodevdd_2kv_tb, cont_drw) +connect(idiodevdd_2kv_c, cont_drw) + +# idiodevdd_4kv +connect(idiodevdd_4kv_e_1x1, idiodevdd_4kv_e) +connect(idiodevdd_4kv_e, idiodevdd_4kv_te) +connect(idiodevdd_4kv_te, cont_drw) +connect(idiodevdd_4kv_b, idiodevdd_4kv_tb) +connect(idiodevdd_4kv_tb, cont_drw) +connect(idiodevdd_4kv_c, cont_drw) + +# idiodevss_2kv +connect(idiodevss_2kv_e_1x1, idiodevss_2kv_e) +connect(idiodevss_2kv_e, cont_drw) +connect(idiodevss_2kv_b, idiodevss_2kv_tb) +connect(idiodevss_2kv_tb, cont_drw) +connect(idiodevss_2kv_c, cont_drw) + +# idiodevss_4kv +connect(idiodevss_4kv_e_1x1, idiodevss_4kv_e) +connect(idiodevss_4kv_e, idiodevss_4kv_te) +connect(idiodevss_4kv_te, cont_drw) +connect(idiodevss_4kv_b, idiodevss_4kv_tb) +connect(idiodevss_4kv_tb, cont_drw) +connect(idiodevss_4kv_c, cont_drw) + +#====================== +# ----- MOSCL-ESD ----- +#====================== + +# nmoscl_2 +connect(nmoscl_2_n, nmoscl_2_n_port) +connect(nmoscl_2_n_port, cont_drw) +connect(nmoscl_2_p, nmoscl_2_p_) +connect(nmoscl_2_p_, ptap) + +# nmoscl_4 +connect(nmoscl_4_n, nmoscl_4_n_port) +connect(nmoscl_4_n_port, cont_drw) +connect(nmoscl_4_p, nmoscl_4_p_) +connect(nmoscl_4_p_, ptap) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_derivations.lvs new file mode 100644 index 00000000..bd78b0ff --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_derivations.lvs @@ -0,0 +1,147 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- MOS-SAB DERIVATIONS ------ +#================================ + +logger.info('Starting ESD DERIVATIONS') + +# General +esd_exclude = nsd_block.join(nsd_drw).join(trans_drw) + .join(emwind_drw).join(emwihv_drw).join(polyres_drw) + .join(extblock_drw).join(res_drw).join(substrate_drw) + .join(ind_drw).join(ind_pin) + +esd_exc_d = gatpoly.join(thickgateox_drw).join(salblock_drw) + .join(esd_exclude) + +idiodevdd_exc = esd_exc_d.join(nwell_holes) +diodevdd_exc = idiodevdd_exc.join(nbulay_drw).join(pwell_block) + +idiodevss_exc = esd_exc_d.join(nwell_drw.not_interacting(nwell_holes)) + .join(pwell_block) +diodevss_exc = idiodevss_exc.join(nbulay_drw) + +nw_diode = nwell_drw.not_interacting(pwell_block) +nw_idiode = nwell_iso.interacting(pwell_block) + +#====================== +# ----- diode-ESD ----- +#====================== + +# diodevdd_2kv +diodevdd_2kv_e = pactiv.and(nw_diode).and(recog_esd).not(diodevdd_exc) +diodevdd_2kv_e_1x1 = diodevdd_2kv_e.middle.sized(0.499.um) +diodevdd_2kv_b_ = nactiv.and(nw_diode).and(recog_esd).not(diodevdd_exc) +diodevdd_2kv_b = diodevdd_2kv_b_.interacting(diodevdd_2kv_b_.extents.interacting(diodevdd_2kv_e, 1, 1)) +diodevdd_2kv_tb = cont_drw.and(diodevdd_2kv_b).not_interacting(diodevdd_2kv_e) +diodevdd_2kv_c_ = pactiv.and(pwell).and(recog_esd).not(diodevdd_exc) +diodevdd_2kv_c = diodevdd_2kv_c_.interacting(diodevdd_2kv_c_.extents.interacting(diodevdd_2kv_b, 1, 1)) + +# diodevdd_4kv +diodevdd_4kv_b = diodevdd_2kv_b_.interacting(diodevdd_2kv_b_.extents.interacting(diodevdd_2kv_e, 2, 2)) +diodevdd_4kv_c = diodevdd_2kv_c_.interacting(diodevdd_2kv_c_.extents.interacting(diodevdd_4kv_b, 1, 1)) +diodevdd_4kv_e = diodevdd_4kv_b.extents.sized(-0.15.um) +diodevdd_4kv_e_1x1 = diodevdd_4kv_e.middle.sized(0.499.um) +diodevdd_4kv_te = cont_drw.and(diodevdd_2kv_e).not_interacting(diodevdd_4kv_b) +diodevdd_4kv_tb = cont_drw.and(diodevdd_4kv_b).not_interacting(diodevdd_2kv_e) + +# diodevss_2kv +diodevss_2kv_e = nactiv.and(pwell).and(recog_esd).not(diodevss_exc) +diodevss_2kv_e_1x1 = diodevss_2kv_e.middle.sized(0.499.um) +diodevss_2kv_b_ = pactiv.and(pwell).and(recog_esd).not(diodevss_exc) +diodevss_2kv_b = diodevss_2kv_b_.interacting(diodevss_2kv_b_.extents.interacting(diodevss_2kv_e, 1, 1)) +diodevss_2kv_tb = cont_drw.and(diodevss_2kv_b).not_interacting(diodevss_2kv_e) +diodevss_2kv_c_ = nactiv.and(nw_diode).and(recog_esd).not(diodevss_exc) +diodevss_2kv_c = diodevss_2kv_c_.interacting(diodevss_2kv_c_.extents.interacting(diodevss_2kv_b, 1, 1)) + +# diodevss_4kv +diodevss_4kv_b = diodevss_2kv_b_.interacting(diodevss_2kv_b_.extents.interacting(diodevss_2kv_e, 2, 2)) +diodevss_4kv_c = diodevss_2kv_c_.interacting(diodevss_2kv_c_.extents.interacting(diodevss_4kv_b, 1, 1)) +diodevss_4kv_e = diodevss_4kv_b.extents.sized(-0.15.um) +diodevss_4kv_e_1x1 = diodevss_4kv_e.middle.sized(0.499.um) +diodevss_4kv_te = cont_drw.and(diodevss_2kv_e).not_interacting(diodevss_4kv_b) +diodevss_4kv_tb = cont_drw.and(diodevss_4kv_b).not_interacting(diodevss_2kv_e) + +#======================= +# ----- idiode-ESD ----- +#======================= + +# idiodevdd_2kv +idiodevdd_2kv_e = pactiv.and(nw_idiode).and(recog_esd).not(idiodevdd_exc) +idiodevdd_2kv_e_1x1 = idiodevdd_2kv_e.middle.sized(0.499.um) +idiodevdd_2kv_b_ = nactiv.and(nw_idiode).and(recog_esd).not(idiodevdd_exc) +idiodevdd_2kv_b = idiodevdd_2kv_b_.interacting(idiodevdd_2kv_b_.extents.interacting(idiodevdd_2kv_e, 1, 1)) +idiodevdd_2kv_tb = cont_drw.and(idiodevdd_2kv_b).not_interacting(idiodevdd_2kv_e) +idiodevdd_2kv_c_ = pactiv.and(pwell).and(recog_esd).not(idiodevdd_exc) +idiodevdd_2kv_c = idiodevdd_2kv_c_.interacting(idiodevdd_2kv_c_.extents.interacting(idiodevdd_2kv_b, 1, 1)) + +# idiodevdd_4kv +idiodevdd_4kv_b = idiodevdd_2kv_b_.interacting(idiodevdd_2kv_b_.extents.interacting(idiodevdd_2kv_e, 2, 2)) +idiodevdd_4kv_c = idiodevdd_2kv_c_.interacting(idiodevdd_2kv_c_.extents.interacting(idiodevdd_4kv_b, 1, 1)) +idiodevdd_4kv_e = idiodevdd_4kv_b.extents.sized(-0.15.um) +idiodevdd_4kv_e_1x1 = idiodevdd_4kv_e.middle.sized(0.499.um) +idiodevdd_4kv_te = cont_drw.and(idiodevdd_2kv_e).not_interacting(idiodevdd_4kv_b) +idiodevdd_4kv_tb = cont_drw.and(idiodevdd_4kv_b).not_interacting(idiodevdd_2kv_e) + +# idiodevss_2kv +idiodevss_2kv_e = nactiv.and(pwell).and(nbulay_drw).and(recog_esd).not(idiodevss_exc) +idiodevss_2kv_e_1x1 = idiodevss_2kv_e.middle.sized(0.499.um) +idiodevss_2kv_b_ = pactiv.and(pwell).and(nbulay_drw).and(recog_esd).not(idiodevss_exc) +idiodevss_2kv_b = idiodevss_2kv_b_.interacting(idiodevss_2kv_b_.extents.interacting(idiodevss_2kv_e, 1, 1)) +idiodevss_2kv_tb = cont_drw.and(idiodevss_2kv_b).not_interacting(idiodevss_2kv_e) +idiodevss_2kv_c_ = nactiv.and(nwell_iso).and(recog_esd).not(idiodevss_exc) +idiodevss_2kv_c = idiodevss_2kv_c_.interacting(idiodevss_2kv_c_.extents.interacting(idiodevss_2kv_b, 1, 1)) + +# idiodevss_4kv +idiodevss_4kv_b = idiodevss_2kv_b_.interacting(idiodevss_2kv_b_.extents.interacting(idiodevss_2kv_e, 2, 2)) +idiodevss_4kv_c = idiodevss_2kv_c_.interacting(idiodevss_2kv_c_.extents.interacting(idiodevss_4kv_b, 1, 1)) +idiodevss_4kv_e = idiodevss_4kv_b.extents.sized(-0.15.um) +idiodevss_4kv_e_1x1 = idiodevss_4kv_e.middle.sized(0.499.um) +idiodevss_4kv_te = cont_drw.and(idiodevss_2kv_e).not_interacting(idiodevss_4kv_b) +idiodevss_4kv_tb = cont_drw.and(idiodevss_4kv_b).not_interacting(idiodevss_2kv_e) + +#====================== +# ----- MOSCL-ESD ----- +#====================== + +nmoscl_exc = esd_exclude.join(pwell_block) + +# nmoscl_2 +nmoscl_2_patt = glob_to_case_insensitive_glob("nmoscl_2") + +gate_moscl = ngate_hv_base.and(salblock_drw).and(nbulay_drw) +nmoscl_2_n_ = recog_esd.interacting(text_drw.texts(nmoscl_2_patt)) +nmoscl_2_n = nmoscl_2_n_.interacting(gate_moscl, 12) +nmoscl_2_n_port = nwell_drw.and(nmoscl_2_n) +nmoscl_2_p_ = ptap.and(nbulay_drw).inside(nmoscl_2_n_) +# Using box with area 1x1 to be used as a reference to (m) +nmoscl_2_p =nmoscl_2_p_.middle(as_boxes).sized(0.499.um) + +# nmoscl_4 +nmoscl_4_patt = glob_to_case_insensitive_glob("nmoscl_4") + +gate_moscl = ngate_hv_base.and(salblock_drw).and(nbulay_drw) +nmoscl_4_n_ = recog_esd.interacting(text_drw.texts(nmoscl_4_patt)) +nmoscl_4_n = nmoscl_4_n_.interacting(gate_moscl, 24) +nmoscl_4_n_port = nwell_drw.and(nmoscl_4_n) +nmoscl_4_p_ = ptap.and(nbulay_drw).inside(nmoscl_4_n_) +# Using box with area 1x1 to be used as a reference to (m) +nmoscl_4_p =nmoscl_4_p_.middle(as_boxes).sized(0.499.um) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_extraction.lvs new file mode 100644 index 00000000..585281ee --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/esd_extraction.lvs @@ -0,0 +1,116 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- MOS-SAB EXTRACTION ------ +#================================ + +logger.info('Starting ESD EXTRACTION') + +#====================== +# ----- diode-ESD ----- +#====================== + +# diodevdd_2kv +logger.info('Extracting diodevdd_2kv ESD device') +extract_devices(bjt3('diodevdd_2kv', Esd3Term), { 'C' => diodevdd_2kv_c.extents, + 'B' => diodevdd_2kv_b.extents, + 'E' => diodevdd_2kv_e_1x1, + 'tC' => diodevdd_2kv_c, + 'tB' => diodevdd_2kv_b, + 'tE' => diodevdd_2kv_e_1x1 }) + +# diodevdd_4kv +logger.info('Extracting diodevdd_4kv ESD device') +extract_devices(bjt3('diodevdd_4kv', Esd3Term), { 'C' => diodevdd_4kv_c.extents, + 'B' => diodevdd_4kv_b.extents, + 'E' => diodevdd_4kv_e_1x1, + 'tC' => diodevdd_4kv_c, + 'tB' => diodevdd_4kv_b, + 'tE' => diodevdd_4kv_e_1x1 }) + +# diodevss_2kv +logger.info('Extracting diodevss_2kv ESD device') +extract_devices(bjt3('diodevss_2kv', Esd3Term), { 'C' => diodevss_2kv_c.extents, + 'B' => diodevss_2kv_b.extents, + 'E' => diodevss_2kv_e_1x1, + 'tC' => diodevss_2kv_c, + 'tB' => diodevss_2kv_b, + 'tE' => diodevss_2kv_e_1x1 }) + +# diodevss_4kv +logger.info('Extracting diodevss_4kv ESD device') +extract_devices(bjt3('diodevss_4kv', Esd3Term), { 'C' => diodevss_4kv_c.extents, + 'B' => diodevss_4kv_b.extents, + 'E' => diodevss_4kv_e_1x1, + 'tC' => diodevss_4kv_c, + 'tB' => diodevss_4kv_b, + 'tE' => diodevss_4kv_e_1x1 }) + +#======================= +# ----- idiode-ESD ----- +#======================= + +# idiodevdd_2kv +logger.info('Extracting idiodevdd_2kv ESD device') +extract_devices(bjt3('idiodevdd_2kv', Esd3Term), { 'C' => idiodevdd_2kv_c.extents, + 'B' => idiodevdd_2kv_b.extents, + 'E' => idiodevdd_2kv_e_1x1, + 'tC' => idiodevdd_2kv_c, + 'tB' => idiodevdd_2kv_b, + 'tE' => idiodevdd_2kv_e_1x1 }) + +# idiodevdd_4kv +logger.info('Extracting idiodevdd_4kv ESD device') +extract_devices(bjt3('idiodevdd_4kv', Esd3Term), { 'C' => idiodevdd_4kv_c.extents, + 'B' => idiodevdd_4kv_b.extents, + 'E' => idiodevdd_4kv_e_1x1, + 'tC' => idiodevdd_4kv_c, + 'tB' => idiodevdd_4kv_b, + 'tE' => idiodevdd_4kv_e_1x1 }) + +# idiodevss_2kv +logger.info('Extracting idiodevss_2kv ESD device') +extract_devices(bjt3('idiodevss_2kv', Esd3Term), { 'C' => idiodevss_2kv_c.extents, + 'B' => idiodevss_2kv_b.extents, + 'E' => idiodevss_2kv_e_1x1, + 'tC' => idiodevss_2kv_c, + 'tB' => idiodevss_2kv_b, + 'tE' => idiodevss_2kv_e_1x1 }) + +# idiodevss_4kv +logger.info('Extracting idiodevss_4kv ESD device') +extract_devices(bjt3('idiodevss_4kv', Esd3Term), { 'C' => idiodevss_4kv_c.extents, + 'B' => idiodevss_4kv_b.extents, + 'E' => idiodevss_4kv_e_1x1, + 'tC' => idiodevss_4kv_c, + 'tB' => idiodevss_4kv_b, + 'tE' => idiodevss_4kv_e_1x1 }) + +#====================== +# ----- MOSCL-ESD ----- +#====================== + +# nmoscl_2 +logger.info('Extracting nmoscl_2 ESD device') +extract_devices(diode('nmoscl_2', Esd2Term), { 'N' => nmoscl_2_n, 'P' => nmoscl_2_p }) + +# nmoscl_4 +logger.info('Extracting nmoscl_4 ESD device') +extract_devices(diode('nmoscl_4', Esd2Term), { 'N' => nmoscl_4_n, 'P' => nmoscl_4_p }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_connections.lvs new file mode 100644 index 00000000..cf3ea988 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_connections.lvs @@ -0,0 +1,64 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- GENERAL CONNECTIONS ----- +#================================ + +logger.info('Starting SG13G2 LVS connectivity setup (Inter-layer)') + +# Inter-layer +connect(pwell_sub, pwell) +connect(pwell, ptap) +connect(nwell_drw, ntap) +connect(ntap, cont_drw) +connect(ptap, cont_drw) +connect(poly_con, cont_drw) +connect(nsd_fet, cont_drw) +connect(psd_fet, cont_drw) +connect(cont_drw, metal1_con) +connect(metal1_con, via1_drw) +connect(via1_drw, metal2_con) +connect(metal2_con, via2_drw) +connect(via2_drw, metal3_con) +connect(metal3_con, via3_drw) +connect(via3_drw, metal4_con) +connect(metal4_con, via4_drw) +connect(via4_drw, metal5_con) +connect(metal5_con, topvia1_n_cap) +connect(topvia1_n_cap, topmetal1_con) +connect(topmetal1_con, topvia2_drw) +connect(topvia2_drw, topmetal2_con) + +# salicide connection +connect(nsd_fet, nsd_ptap_abutt) +connect(nsd_ptap_abutt, ptap) +connect(psd_fet, psd_ntap_abutt) +connect(psd_ntap_abutt, ntap) + +# Attaching labels +connect(metal1_con, metal1_text) +connect(metal2_con, metal2_text) +connect(metal3_con, metal3_text) +connect(metal4_con, metal4_text) +connect(metal5_con, metal5_text) +connect(topmetal1_con, topmetal1_text) +connect(topmetal2_con, topmetal2_text) + +logger.info('Starting SG13G2 LVS connectivity setup (Global)') diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_derivations.lvs new file mode 100644 index 00000000..d6780f24 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/general_derivations.lvs @@ -0,0 +1,104 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================== +# ------ GENERAL DERIVATIONS ------ +#================================== + +logger.info('Starting general LVS derivations') + +#=== Global Layers === +# === CHIP === +CHIP = case $run_mode +when 'deep' + extent('*') +else + #=== FLAT MODE === + extent.sized(0.0) +end + +# === General Derivations === +# nwell +nwell_iso = nwell_drw.and(nbulay_drw) +nwell_holes = nwell_drw.holes.not(nwell_drw) + +# pwell +pwell_allowed = CHIP.not(pwell_block) +digisub_gap = digisub_drw.not(digisub_drw.sized(-1.nm)) +pwell = pwell_allowed.not(nwell_drw).not(digisub_gap) + +# General pwell +pwell_sub = pwell_allowed.not(digisub_drw).not(nbulay_drw.interacting(nwell_holes)) + +# n & p activ +nactiv = activ.not(psd_drw.join(nsd_block)) +pactiv = activ.and(psd_drw) + +# res/cap exclusion +res_mk = polyres_drw.join(res_drw) +poly_con = gatpoly.not(res_mk) +metal1_con = metal1.not(metal1_res) +metal2_con = metal2.not(metal2_res) +metal3_con = metal3.not(metal3_res) +metal4_con = metal4.not(metal4_res) +metal5_con = metal5.not(metal5_res) +topmetal1_con = topmetal1.not(topmetal1_res).not(ind_drw) +topmetal2_con = topmetal2.not(topmetal2_res).not(ind_drw) + +# Gate FETs +tgate = gatpoly.and(activ).not(res_mk) +ngate = nactiv.and(tgate) +pgate = pactiv.and(tgate) +ngate_lv_base = ngate.not(thickgateox_drw) +pgate_lv_base = pgate.not(thickgateox_drw) +ngate_hv_base = ngate.and(thickgateox_drw) +pgate_hv_base = pgate.and(thickgateox_drw) + +# S/D FETs +nsd_fet = nactiv.not(nwell_drw).interacting(ngate).not(ngate).not_interacting(res_mk) +psd_fet = pactiv.and(nwell_drw).interacting(pgate).not(pgate).not_interacting(res_mk) + + +# n1/p1 taps labels +well_patt = glob_to_case_insensitive_glob("well") +sub_patt = glob_to_case_insensitive_glob("sub!") + +ntap1_lbl = text_drw.texts(well_patt) +ntap1_mk = nwell_drw.interacting(ntap1_lbl) + +ptap1_lbl = text_drw.texts(sub_patt) +ptap1_mk = substrate_drw.and(pwell).interacting(ptap1_lbl) + +# n & p taps (short connections) +ntap = nactiv.and(nwell_drw).not(recog_diode).not(gatpoly).not(ntap1_mk) +ptap = pactiv.and(pwell).not(ptap1_mk).not(recog_diode).not(gatpoly) +ptap_holes = ptap.holes +ntap_holes = ntap.holes + +# S/D (salicide) +nsd_sal = nsd_fet.not(salblock_drw) +psd_sal = psd_fet.not(salblock_drw) + +# n & p taps (salicide) +ntap_sal = ntap.not(salblock_drw) +ptap_sal = ptap.not(salblock_drw) + +# n/p SD abutted with n/p taps (salicide) +nsd_ptap_abutt = nsd_sal.edges.and(ptap_sal.edges).extended(:in => 1.nm, :out => 1.nm) +psd_ntap_abutt = psd_sal.edges.and(ntap_sal.edges).extended(:in => 1.nm, :out => 1.nm) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/globals.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/globals.lvs new file mode 100644 index 00000000..2f49defc --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/globals.lvs @@ -0,0 +1,68 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=========================================== +# --------------- Varaibles ---------------- +#=========================================== + +# Prefix for each device +PREFIX_MAP = { + 'sg13_lv_nmos' => 'M', + 'sg13_hv_nmos' => 'M', + 'sg13_lv_pmos' => 'M', + 'sg13_hv_pmos' => 'M', + 'rfnmos' => 'M', + 'rfnmosHV' => 'M', + 'rfpmos' => 'M', + 'rfpmosHV' => 'M', + 'npn13G2' => 'Q', + 'npn13G2L' => 'Q', + 'npn13G2V' => 'Q', + 'pnpMPA' => 'Q', + 'dantenna' => 'D', + 'dpantenna' => 'D', + 'schottky_nbl1' => 'D', + 'rsil' => 'R', + 'rppd' => 'R', + 'rhigh' => 'R', + 'lvsres' => 'R', + 'SVaricap' => 'C', + 'cap_cmim' => 'C', + 'rfcmim' => 'C', + 'diodevss_4kv' => 'D', + 'diodevss_2kv' => 'D', + 'diodevdd_4kv' => 'D', + 'diodevdd_2kv' => 'D', + 'idiodevss_4kv' => 'D', + 'idiodevss_2kv' => 'D', + 'idiodevdd_4kv' => 'D', + 'idiodevdd_2kv' => 'D', + 'nmoscl_2' => 'D', + 'nmoscl_4' => 'D', + 'ptap1' => 'R', + 'ntap1' => 'R', + 'inductor2' => 'L', + 'inductor3' => 'L' +}.freeze + +# Prefix for devices will be customized +CUSTOM_READER = %w[M C R Q L D].freeze + +# List of poly-resistors +RES_DEV = ['rsil', 'rppd', 'rhigh'] diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_connections.lvs new file mode 100644 index 00000000..3ad62294 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_connections.lvs @@ -0,0 +1,37 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=================================== +# ------ Inductor CONNECTIONS ------ +#=================================== + +logger.info('Starting Inductor CONNECTIONS') + +# ind2 +connect(ind2_ports, ind_pin) +connect(ind_pin, ind_text) +connect(ind_pin, topmetal2_con) +connect(ind2_sub, pwell) +connect(ind2_sub, nwell_drw) + +# ind3 +connect(ind3_ports, ind_pin) +connect(ind_pin, topmetal1_con) +connect(ind3_sub, pwell) +connect(ind3_sub, nwell_drw) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_derivations.lvs new file mode 100644 index 00000000..079ffadd --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_derivations.lvs @@ -0,0 +1,66 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=============================== +# ---- Inductor DERIVATIONS ---- +#=============================== + +logger.info('Starting Inductor DERIVATIONS') + +ind_exc = gatpoly.join(nsd_drw).join(nbulay_drw) + .join(thickgateox_drw).join(emwind_drw).join(emwihv_drw) + .join(salblock_drw).join(polyres_drw).join(mim_drw) + .join(extblock_drw).join(res_drw).join(activ_mask) + .join(recog_diode).join(recog_esd).join(substrate_drw) + +# General +la_patt = glob_to_case_insensitive_glob("LA") +lb_patt = glob_to_case_insensitive_glob("LB") +lc_patt = glob_to_case_insensitive_glob("LC") +ind2_patt = glob_to_case_insensitive_glob("inductor2*") +ind3_patt = glob_to_case_insensitive_glob("inductor3*") + +ind_edges = ind_drw.edges +ind_core_ = topmetal2.join(topmetal1).and(ind_drw).merged.not(ind_exc) +ind_ports_ = ind_pin.and(ind_core_).interacting(ind_edges) +ind_port_la = ind_ports_.interacting(ind_text.texts(la_patt)) +ind_la_tm1 = ind_port_la.and(topmetal1) +ind_port_lb = ind_ports_.interacting(ind_text.texts(lb_patt)) +ind_lb_tm1 = ind_port_lb.and(topmetal1) +ind_port_lc = ind_ports_.interacting(ind_text.texts(lc_patt)) +ind_lc_tm2 = ind_port_lc.and(topmetal2) + +# inductor2 +ind2_ports = ind_port_la.join(ind_port_lb) +ind2_core = ind_core_.interacting(ind_port_la, 1).interacting(ind_port_lb, 1) +ind2_mk_ = ind_drw.interacting(text_drw.texts(ind2_patt)) +ind2_mk = ind2_mk_.interacting(ind2_core).interacting(ind2_ports).not_interacting(ind_port_lc) +ind2_sub1 = pwell.and(ind_drw).interacting(ind2_core) +ind2_sub2 = pwell_block.and(ind_drw).interacting(ind2_core).sized(1.nm) +ind2_well = nwell_drw.and(ind_drw).interacting(ind2_core).sized(-1.nm) +ind2_sub = ind2_sub1.join(ind2_sub2).join(ind2_well) +# inductor3 +ind3_ports = ind_la_tm1.join(ind_lb_tm1).join(ind_lc_tm2) +ind3_core = ind_core_.interacting(ind_lb_tm1, 1).interacting(ind_lb_tm1, 1).interacting(ind_lc_tm2, 1) +ind3_mk_ = ind_drw.interacting(text_drw.texts(ind3_patt)) +ind3_mk = ind3_mk_.interacting(ind3_core).interacting(ind3_ports) +ind3_sub1 = pwell.and(ind_drw).interacting(ind3_core) +ind3_sub2 = pwell_block.and(ind_drw).interacting(ind3_core).sized(1.nm) +ind3_well = nwell_drw.and(ind_drw).interacting(ind3_core).sized(-1.nm) +ind3_sub = ind3_sub1.join(ind3_sub2).join(ind3_well) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_extraction.lvs new file mode 100644 index 00000000..3d2b6b49 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/ind_extraction.lvs @@ -0,0 +1,44 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- Inductor EXTRACTION ----- +#================================ + +logger.info('Starting Inductor EXTRACTION') + +# ind2 +logger.info('Extracting Inductor2 device') +extract_devices(GeneralNTerminalExtractor.new('inductor', 2), { + 'core' => ind2_core, + 'ports' => ind2_ports, + 'meas_mk' => topmetal2.and(ind2_core), + 'dev_mk' => ind2_mk, + 'sub_mk' => ind2_sub + }) + +# ind3 +logger.info('Extracting Inductor3 device') +extract_devices(GeneralNTerminalExtractor.new('inductor3', 3), { + 'core' => ind3_core, + 'ports' => ind3_ports, + 'meas_mk' => topmetal2.and(ind3_core), + 'dev_mk' => ind3_mk, + 'sub_mk' => ind3_sub + }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/layers_definitions.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/layers_definitions.lvs new file mode 100644 index 00000000..e6cf6ba5 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/layers_definitions.lvs @@ -0,0 +1,1856 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================================ +#------------- LAYERS DEFINITIONS --------------- +#================================================ + +polygons_count = 0 +logger.info('Read in polygons from layers.') + +def get_polygons(lay_no, lay_dt) + if $run_mode == 'deep' + polygons(lay_no, lay_dt) + else + polygons(lay_no, lay_dt).merged + end +end + +nodrc_drw = get_polygons(62, 0) +count = nodrc_drw.count() +logger.info("nodrc_drw has #{count} polygons") +polygons_count += count + +activ_drw = get_polygons(1, 0) +activ_drw = activ_drw.not(nodrc_drw) +count = activ_drw.count() +logger.info("activ_drw has #{count} polygons") +polygons_count += count + +activ_filler = get_polygons(1, 22) +activ_filler = activ_filler.not(nodrc_drw) +count = activ_filler.count() +logger.info("activ_filler has #{count} polygons") +polygons_count += count + +# activ org +activ = activ_drw.join(activ_filler) +count = activ.count() +logger.info("activ has #{count} polygons") + +activ_pin = get_polygons(1, 2) +activ_pin = activ_pin.not(nodrc_drw) +count = activ_pin.count() +logger.info("activ_pin has #{count} polygons") +polygons_count += count + +activ_mask = get_polygons(1, 20) +activ_mask = activ_mask.not(nodrc_drw) +count = activ_mask.count() +logger.info("activ_mask has #{count} polygons") +polygons_count += count + +activ_nofill = get_polygons(1, 23) +activ_nofill = activ_nofill.not(nodrc_drw) +count = activ_nofill.count() +logger.info("activ_nofill has #{count} polygons") +polygons_count += count + +activ_OPC = get_polygons(1, 26) +activ_OPC = activ_OPC.not(nodrc_drw) +count = activ_OPC.count() +logger.info("activ_OPC has #{count} polygons") +polygons_count += count + +activ_iOPC = get_polygons(1, 27) +activ_iOPC = activ_iOPC.not(nodrc_drw) +count = activ_iOPC.count() +logger.info("activ_iOPC has #{count} polygons") +polygons_count += count + +activ_noqrc = get_polygons(1, 28) +activ_noqrc = activ_noqrc.not(nodrc_drw) +count = activ_noqrc.count() +logger.info("activ_noqrc has #{count} polygons") +polygons_count += count + +biwind_drw = get_polygons(3, 0) +biwind_drw = biwind_drw.not(nodrc_drw) +count = biwind_drw.count() +logger.info("biwind_drw has #{count} polygons") +polygons_count += count + +biwind_OPC = get_polygons(3, 26) +biwind_OPC = biwind_OPC.not(nodrc_drw) +count = biwind_OPC.count() +logger.info("biwind_OPC has #{count} polygons") +polygons_count += count + +gatpoly_drw = get_polygons(5, 0) +gatpoly_drw = gatpoly_drw.not(nodrc_drw) +count = gatpoly_drw.count() +logger.info("gatpoly_drw has #{count} polygons") +polygons_count += count + +gatpoly_filler = get_polygons(5, 22) +gatpoly_filler = gatpoly_filler.not(nodrc_drw) +count = gatpoly_filler.count() +logger.info("gatpoly_filler has #{count} polygons") +polygons_count += count + +# gatpoly org +gatpoly = gatpoly_drw.join(gatpoly_filler) +count = gatpoly.count() +logger.info("gatpoly has #{count} polygons") + +gatpoly_pin = get_polygons(5, 2) +gatpoly_pin = gatpoly_pin.not(nodrc_drw) +count = gatpoly_pin.count() +logger.info("gatpoly_pin has #{count} polygons") +polygons_count += count + +gatpoly_nofill = get_polygons(5, 23) +gatpoly_nofill = gatpoly_nofill.not(nodrc_drw) +count = gatpoly_nofill.count() +logger.info("gatpoly_nofill has #{count} polygons") +polygons_count += count + +gatpoly_OPC = get_polygons(5, 26) +gatpoly_OPC = gatpoly_OPC.not(nodrc_drw) +count = gatpoly_OPC.count() +logger.info("gatpoly_OPC has #{count} polygons") +polygons_count += count + +gatpoly_iOPC = get_polygons(5, 27) +gatpoly_iOPC = gatpoly_iOPC.not(nodrc_drw) +count = gatpoly_iOPC.count() +logger.info("gatpoly_iOPC has #{count} polygons") +polygons_count += count + +gatpoly_noqrc = get_polygons(5, 28) +gatpoly_noqrc = gatpoly_noqrc.not(nodrc_drw) +count = gatpoly_noqrc.count() +logger.info("gatpoly_noqrc has #{count} polygons") +polygons_count += count + +cont_drw = get_polygons(6, 0) +cont_drw = cont_drw.not(nodrc_drw) +count = cont_drw.count() +logger.info("cont_drw has #{count} polygons") +polygons_count += count + +cont_OPC = get_polygons(6, 26) +cont_OPC = cont_OPC.not(nodrc_drw) +count = cont_OPC.count() +logger.info("cont_OPC has #{count} polygons") +polygons_count += count + +nsd_drw = get_polygons(7, 0) +nsd_drw = nsd_drw.not(nodrc_drw) +count = nsd_drw.count() +logger.info("nsd_drw has #{count} polygons") +polygons_count += count + +nsd_block = get_polygons(7, 21) +nsd_block = nsd_block.not(nodrc_drw) +count = nsd_block.count() +logger.info("nsd_block has #{count} polygons") +polygons_count += count + +metal1_drw = get_polygons(8, 0) +metal1_drw = metal1_drw.not(nodrc_drw) +count = metal1_drw.count() +logger.info("metal1_drw has #{count} polygons") +polygons_count += count + +metal1_filler = get_polygons(8, 22) +metal1_filler = metal1_filler.not(nodrc_drw) +count = metal1_filler.count() +logger.info("metal1_filler has #{count} polygons") +polygons_count += count + +metal1_slit = get_polygons(8, 24) +metal1_slit = metal1_slit.not(nodrc_drw) +count = metal1_slit.count() +logger.info("metal1_slit has #{count} polygons") +polygons_count += count + +# metal1 org +metal1 = metal1_drw.join(metal1_filler).not(metal1_slit) +count = metal1.count() +logger.info("metal1 has #{count} polygons") + +metal1_pin = get_polygons(8, 2) +metal1_pin = metal1_pin.not(nodrc_drw) +count = metal1_pin.count() +logger.info("metal1_pin has #{count} polygons") +polygons_count += count + +metal1_mask = get_polygons(8, 20) +metal1_mask = metal1_mask.not(nodrc_drw) +count = metal1_mask.count() +logger.info("metal1_mask has #{count} polygons") +polygons_count += count + +metal1_nofill = get_polygons(8, 23) +metal1_nofill = metal1_nofill.not(nodrc_drw) +count = metal1_nofill.count() +logger.info("metal1_nofill has #{count} polygons") +polygons_count += count + +metal1_text = labels(8, 25) +metal1_text = metal1_text.not(nodrc_drw) +count = metal1_text.count() +logger.info("metal1_text has #{count} polygons") +polygons_count += count + +metal1_OPC = get_polygons(8, 26) +metal1_OPC = metal1_OPC.not(nodrc_drw) +count = metal1_OPC.count() +logger.info("metal1_OPC has #{count} polygons") +polygons_count += count + +metal1_noqrc = get_polygons(8, 28) +metal1_noqrc = metal1_noqrc.not(nodrc_drw) +count = metal1_noqrc.count() +logger.info("metal1_noqrc has #{count} polygons") +polygons_count += count + +metal1_res = get_polygons(8, 29) +metal1_res = metal1_res.not(nodrc_drw) +count = metal1_res.count() +logger.info("metal1_res has #{count} polygons") +polygons_count += count + +metal1_iprobe = get_polygons(8, 33) +metal1_iprobe = metal1_iprobe.not(nodrc_drw) +count = metal1_iprobe.count() +logger.info("metal1_iprobe has #{count} polygons") +polygons_count += count + +metal1_diffprb = get_polygons(8, 34) +metal1_diffprb = metal1_diffprb.not(nodrc_drw) +count = metal1_diffprb.count() +logger.info("metal1_diffprb has #{count} polygons") +polygons_count += count + +passiv_drw = get_polygons(9, 0) +passiv_drw = passiv_drw.not(nodrc_drw) +count = passiv_drw.count() +logger.info("passiv_drw has #{count} polygons") +polygons_count += count + +passiv_pin = get_polygons(9, 2) +passiv_pin = passiv_pin.not(nodrc_drw) +count = passiv_pin.count() +logger.info("passiv_pin has #{count} polygons") +polygons_count += count + +passiv_sbump = get_polygons(9, 36) +passiv_sbump = passiv_sbump.not(nodrc_drw) +count = passiv_sbump.count() +logger.info("passiv_sbump has #{count} polygons") +polygons_count += count + +passiv_pillar = get_polygons(9, 35) +passiv_pillar = passiv_pillar.not(nodrc_drw) +count = passiv_pillar.count() +logger.info("passiv_pillar has #{count} polygons") +polygons_count += count + +passiv_pdl = get_polygons(9, 40) +passiv_pdl = passiv_pdl.not(nodrc_drw) +count = passiv_pdl.count() +logger.info("passiv_pdl has #{count} polygons") +polygons_count += count + +metal2_drw = get_polygons(10, 0) +metal2_drw = metal2_drw.not(nodrc_drw) +count = metal2_drw.count() +logger.info("metal2_drw has #{count} polygons") +polygons_count += count + +metal2_filler = get_polygons(10, 22) +metal2_filler = metal2_filler.not(nodrc_drw) +count = metal2_filler.count() +logger.info("metal2_filler has #{count} polygons") +polygons_count += count + +metal2_slit = get_polygons(10, 24) +metal2_slit = metal2_slit.not(nodrc_drw) +count = metal2_slit.count() +logger.info("metal2_slit has #{count} polygons") +polygons_count += count + +# metal2 org +metal2 = metal2_drw.join(metal2_filler).not(metal2_slit) +count = metal2.count() +logger.info("metal2 has #{count} polygons") + +metal2_pin = get_polygons(10, 2) +metal2_pin = metal2_pin.not(nodrc_drw) +count = metal2_pin.count() +logger.info("metal2_pin has #{count} polygons") +polygons_count += count + +metal2_mask = get_polygons(10, 20) +metal2_mask = metal2_mask.not(nodrc_drw) +count = metal2_mask.count() +logger.info("metal2_mask has #{count} polygons") +polygons_count += count + +metal2_nofill = get_polygons(10, 23) +metal2_nofill = metal2_nofill.not(nodrc_drw) +count = metal2_nofill.count() +logger.info("metal2_nofill has #{count} polygons") +polygons_count += count + +metal2_text = labels(10, 25) +metal2_text = metal2_text.not(nodrc_drw) +count = metal2_text.count() +logger.info("metal2_text has #{count} polygons") +polygons_count += count + +metal2_OPC = get_polygons(10, 26) +metal2_OPC = metal2_OPC.not(nodrc_drw) +count = metal2_OPC.count() +logger.info("metal2_OPC has #{count} polygons") +polygons_count += count + +metal2_noqrc = get_polygons(10, 28) +metal2_noqrc = metal2_noqrc.not(nodrc_drw) +count = metal2_noqrc.count() +logger.info("metal2_noqrc has #{count} polygons") +polygons_count += count + +metal2_res = get_polygons(10, 29) +metal2_res = metal2_res.not(nodrc_drw) +count = metal2_res.count() +logger.info("metal2_res has #{count} polygons") +polygons_count += count + +metal2_iprobe = get_polygons(10, 33) +metal2_iprobe = metal2_iprobe.not(nodrc_drw) +count = metal2_iprobe.count() +logger.info("metal2_iprobe has #{count} polygons") +polygons_count += count + +metal2_diffprb = get_polygons(10, 34) +metal2_diffprb = metal2_diffprb.not(nodrc_drw) +count = metal2_diffprb.count() +logger.info("metal2_diffprb has #{count} polygons") +polygons_count += count + +baspoly_drw = get_polygons(13, 0) +baspoly_drw = baspoly_drw.not(nodrc_drw) +count = baspoly_drw.count() +logger.info("baspoly_drw has #{count} polygons") +polygons_count += count + +baspoly_pin = get_polygons(13, 2) +baspoly_pin = baspoly_pin.not(nodrc_drw) +count = baspoly_pin.count() +logger.info("baspoly_pin has #{count} polygons") +polygons_count += count + +psd_drw = get_polygons(14, 0) +psd_drw = psd_drw.not(nodrc_drw) +count = psd_drw.count() +logger.info("psd_drw has #{count} polygons") +polygons_count += count + +nldb_drw = get_polygons(15, 0) +nldb_drw = nldb_drw.not(nodrc_drw) +count = nldb_drw.count() +logger.info("nldb_drw has #{count} polygons") +polygons_count += count + +digibnd_drw = get_polygons(16, 0) +digibnd_drw = digibnd_drw.not(nodrc_drw) +count = digibnd_drw.count() +logger.info("digibnd_drw has #{count} polygons") +polygons_count += count + +via1_drw = get_polygons(19, 0) +via1_drw = via1_drw.not(nodrc_drw) +count = via1_drw.count() +logger.info("via1_drw has #{count} polygons") +polygons_count += count + +backmetal1_drw = get_polygons(20, 0) +backmetal1_drw = backmetal1_drw.not(nodrc_drw) +count = backmetal1_drw.count() +logger.info("backmetal1_drw has #{count} polygons") +polygons_count += count + +backmetal1_pin = get_polygons(20, 2) +backmetal1_pin = backmetal1_pin.not(nodrc_drw) +count = backmetal1_pin.count() +logger.info("backmetal1_pin has #{count} polygons") +polygons_count += count + +backmetal1_mask = get_polygons(20, 20) +backmetal1_mask = backmetal1_mask.not(nodrc_drw) +count = backmetal1_mask.count() +logger.info("backmetal1_mask has #{count} polygons") +polygons_count += count + +backmetal1_filler = get_polygons(20, 22) +backmetal1_filler = backmetal1_filler.not(nodrc_drw) +count = backmetal1_filler.count() +logger.info("backmetal1_filler has #{count} polygons") +polygons_count += count + +backmetal1_nofill = get_polygons(20, 23) +backmetal1_nofill = backmetal1_nofill.not(nodrc_drw) +count = backmetal1_nofill.count() +logger.info("backmetal1_nofill has #{count} polygons") +polygons_count += count + +backmetal1_slit = get_polygons(20, 24) +backmetal1_slit = backmetal1_slit.not(nodrc_drw) +count = backmetal1_slit.count() +logger.info("backmetal1_slit has #{count} polygons") +polygons_count += count + +backmetal1_text = labels(20, 25) +backmetal1_text = backmetal1_text.not(nodrc_drw) +count = backmetal1_text.count() +logger.info("backmetal1_text has #{count} polygons") +polygons_count += count + +backmetal1_OPC = get_polygons(20, 26) +backmetal1_OPC = backmetal1_OPC.not(nodrc_drw) +count = backmetal1_OPC.count() +logger.info("backmetal1_OPC has #{count} polygons") +polygons_count += count + +backmetal1_noqrc = get_polygons(20, 28) +backmetal1_noqrc = backmetal1_noqrc.not(nodrc_drw) +count = backmetal1_noqrc.count() +logger.info("backmetal1_noqrc has #{count} polygons") +polygons_count += count + +backmetal1_res = get_polygons(20, 29) +backmetal1_res = backmetal1_res.not(nodrc_drw) +count = backmetal1_res.count() +logger.info("backmetal1_res has #{count} polygons") +polygons_count += count + +backmetal1_iprobe = get_polygons(20, 33) +backmetal1_iprobe = backmetal1_iprobe.not(nodrc_drw) +count = backmetal1_iprobe.count() +logger.info("backmetal1_iprobe has #{count} polygons") +polygons_count += count + +backmetal1_diffprb = get_polygons(20, 34) +backmetal1_diffprb = backmetal1_diffprb.not(nodrc_drw) +count = backmetal1_diffprb.count() +logger.info("backmetal1_diffprb has #{count} polygons") +polygons_count += count + +backpassiv_drw = get_polygons(23, 0) +backpassiv_drw = backpassiv_drw.not(nodrc_drw) +count = backpassiv_drw.count() +logger.info("backpassiv_drw has #{count} polygons") +polygons_count += count + +res_drw = get_polygons(24, 0) +res_drw = res_drw.not(nodrc_drw) +count = res_drw.count() +logger.info("res_drw has #{count} polygons") +polygons_count += count + +sram_drw = get_polygons(25, 0) +sram_drw = sram_drw.not(nodrc_drw) +count = sram_drw.count() +logger.info("sram_drw has #{count} polygons") +polygons_count += count + +trans_drw = get_polygons(26, 0) +trans_drw = trans_drw.not(nodrc_drw) +count = trans_drw.count() +logger.info("trans_drw has #{count} polygons") +polygons_count += count + +ind_drw = get_polygons(27, 0) +ind_drw = ind_drw.not(nodrc_drw) +count = ind_drw.count() +logger.info("ind_drw has #{count} polygons") +polygons_count += count + +ind_pin = get_polygons(27, 2) +ind_pin = ind_pin.not(nodrc_drw) +count = ind_pin.count() +logger.info("ind_pin has #{count} polygons") +polygons_count += count + +ind_text = labels(27, 25) +ind_text = ind_text.not(nodrc_drw) +count = ind_text.count() +logger.info("ind_text has #{count} polygons") +polygons_count += count + +salblock_drw = get_polygons(28, 0) +salblock_drw = salblock_drw.not(nodrc_drw) +count = salblock_drw.count() +logger.info("salblock_drw has #{count} polygons") +polygons_count += count + +via2_drw = get_polygons(29, 0) +via2_drw = via2_drw.not(nodrc_drw) +count = via2_drw.count() +logger.info("via2_drw has #{count} polygons") +polygons_count += count + +metal3_drw = get_polygons(30, 0) +metal3_drw = metal3_drw.not(nodrc_drw) +count = metal3_drw.count() +logger.info("metal3_drw has #{count} polygons") +polygons_count += count + +metal3_filler = get_polygons(30, 22) +metal3_filler = metal3_filler.not(nodrc_drw) +count = metal3_filler.count() +logger.info("metal3_filler has #{count} polygons") +polygons_count += count + +metal3_slit = get_polygons(30, 24) +metal3_slit = metal3_slit.not(nodrc_drw) +count = metal3_slit.count() +logger.info("metal3_slit has #{count} polygons") +polygons_count += count + +# metal3 org +metal3 = metal3_drw.join(metal3_filler).not(metal3_slit) +count = metal3.count() +logger.info("metal3 has #{count} polygons") + +metal3_pin = get_polygons(30, 2) +metal3_pin = metal3_pin.not(nodrc_drw) +count = metal3_pin.count() +logger.info("metal3_pin has #{count} polygons") +polygons_count += count + +metal3_mask = get_polygons(30, 20) +metal3_mask = metal3_mask.not(nodrc_drw) +count = metal3_mask.count() +logger.info("metal3_mask has #{count} polygons") +polygons_count += count + +metal3_nofill = get_polygons(30, 23) +metal3_nofill = metal3_nofill.not(nodrc_drw) +count = metal3_nofill.count() +logger.info("metal3_nofill has #{count} polygons") +polygons_count += count + +metal3_text = labels(30, 25) +metal3_text = metal3_text.not(nodrc_drw) +count = metal3_text.count() +logger.info("metal3_text has #{count} polygons") +polygons_count += count + +metal3_OPC = get_polygons(30, 26) +metal3_OPC = metal3_OPC.not(nodrc_drw) +count = metal3_OPC.count() +logger.info("metal3_OPC has #{count} polygons") +polygons_count += count + +metal3_noqrc = get_polygons(30, 28) +metal3_noqrc = metal3_noqrc.not(nodrc_drw) +count = metal3_noqrc.count() +logger.info("metal3_noqrc has #{count} polygons") +polygons_count += count + +metal3_res = get_polygons(30, 29) +metal3_res = metal3_res.not(nodrc_drw) +count = metal3_res.count() +logger.info("metal3_res has #{count} polygons") +polygons_count += count + +metal3_iprobe = get_polygons(30, 33) +metal3_iprobe = metal3_iprobe.not(nodrc_drw) +count = metal3_iprobe.count() +logger.info("metal3_iprobe has #{count} polygons") +polygons_count += count + +metal3_diffprb = get_polygons(30, 34) +metal3_diffprb = metal3_diffprb.not(nodrc_drw) +count = metal3_diffprb.count() +logger.info("metal3_diffprb has #{count} polygons") +polygons_count += count + +nwell_drw = get_polygons(31, 0) +nwell_drw = nwell_drw.not(nodrc_drw) +count = nwell_drw.count() +logger.info("nwell_drw has #{count} polygons") +polygons_count += count + +nwell_pin = get_polygons(31, 2) +nwell_pin = nwell_pin.not(nodrc_drw) +count = nwell_pin.count() +logger.info("nwell_pin has #{count} polygons") +polygons_count += count + +nbulay_drw = get_polygons(32, 0) +nbulay_drw = nbulay_drw.not(nodrc_drw) +count = nbulay_drw.count() +logger.info("nbulay_drw has #{count} polygons") +polygons_count += count + +nbulay_pin = get_polygons(32, 2) +nbulay_pin = nbulay_pin.not(nodrc_drw) +count = nbulay_pin.count() +logger.info("nbulay_pin has #{count} polygons") +polygons_count += count + +nbulay_block = get_polygons(32, 21) +nbulay_block = nbulay_block.not(nodrc_drw) +count = nbulay_block.count() +logger.info("nbulay_block has #{count} polygons") +polygons_count += count + +emwind_drw = get_polygons(33, 0) +emwind_drw = emwind_drw.not(nodrc_drw) +count = emwind_drw.count() +logger.info("emwind_drw has #{count} polygons") +polygons_count += count + +emwind_OPC = get_polygons(33, 26) +emwind_OPC = emwind_OPC.not(nodrc_drw) +count = emwind_OPC.count() +logger.info("emwind_OPC has #{count} polygons") +polygons_count += count + +deepco_drw = get_polygons(35, 0) +deepco_drw = deepco_drw.not(nodrc_drw) +count = deepco_drw.count() +logger.info("deepco_drw has #{count} polygons") +polygons_count += count + +mim_drw = get_polygons(36, 0) +mim_drw = mim_drw.not(nodrc_drw) +count = mim_drw.count() +logger.info("mim_drw has #{count} polygons") +polygons_count += count + +edgeseal_drw = get_polygons(39, 0) +edgeseal_drw = edgeseal_drw.not(nodrc_drw) +count = edgeseal_drw.count() +logger.info("edgeseal_drw has #{count} polygons") +polygons_count += count + +substrate_drw = get_polygons(40, 0) +substrate_drw = substrate_drw.not(nodrc_drw) +count = substrate_drw.count() +logger.info("substrate_drw has #{count} polygons") +polygons_count += count + +substrate_text = labels(40, 25) +substrate_text = substrate_text.not(nodrc_drw) +count = substrate_text.count() +logger.info("substrate_text has #{count} polygons") +polygons_count += count + +dfpad_drw = get_polygons(41, 0) +dfpad_drw = dfpad_drw.not(nodrc_drw) +count = dfpad_drw.count() +logger.info("dfpad_drw has #{count} polygons") +polygons_count += count + +dfpad_pillar = get_polygons(41, 35) +dfpad_pillar = dfpad_pillar.not(nodrc_drw) +count = dfpad_pillar.count() +logger.info("dfpad_pillar has #{count} polygons") +polygons_count += count + +dfpad_sbump = get_polygons(41, 36) +dfpad_sbump = dfpad_sbump.not(nodrc_drw) +count = dfpad_sbump.count() +logger.info("dfpad_sbump has #{count} polygons") +polygons_count += count + +thickgateox_drw = get_polygons(44, 0) +thickgateox_drw = thickgateox_drw.not(nodrc_drw) +count = thickgateox_drw.count() +logger.info("thickgateox_drw has #{count} polygons") +polygons_count += count + +pldb_drw = get_polygons(45, 0) +pldb_drw = pldb_drw.not(nodrc_drw) +count = pldb_drw.count() +logger.info("pldb_drw has #{count} polygons") +polygons_count += count + +pwell_drw = get_polygons(46, 0) +pwell_drw = pwell_drw.not(nodrc_drw) +count = pwell_drw.count() +logger.info("pwell_drw has #{count} polygons") +polygons_count += count + +pwell_pin = get_polygons(46, 2) +pwell_pin = pwell_pin.not(nodrc_drw) +count = pwell_pin.count() +logger.info("pwell_pin has #{count} polygons") +polygons_count += count + +pwell_block = get_polygons(46, 21) +pwell_block = pwell_block.not(nodrc_drw) +count = pwell_block.count() +logger.info("pwell_block has #{count} polygons") +polygons_count += count + +ic_drw = get_polygons(48, 0) +ic_drw = ic_drw.not(nodrc_drw) +count = ic_drw.count() +logger.info("ic_drw has #{count} polygons") +polygons_count += count + +via3_drw = get_polygons(49, 0) +via3_drw = via3_drw.not(nodrc_drw) +count = via3_drw.count() +logger.info("via3_drw has #{count} polygons") +polygons_count += count + +metal4_drw = get_polygons(50, 0) +metal4_drw = metal4_drw.not(nodrc_drw) +count = metal4_drw.count() +logger.info("metal4_drw has #{count} polygons") +polygons_count += count + +metal4_filler = get_polygons(50, 22) +metal4_filler = metal4_filler.not(nodrc_drw) +count = metal4_filler.count() +logger.info("metal4_filler has #{count} polygons") +polygons_count += count + +metal4_slit = get_polygons(50, 24) +metal4_slit = metal4_slit.not(nodrc_drw) +count = metal4_slit.count() +logger.info("metal4_slit has #{count} polygons") +polygons_count += count + +# metal4 org +metal4 = metal4_drw.join(metal4_filler).not(metal4_slit) +count = metal4.count() +logger.info("metal4 has #{count} polygons") + +metal4_pin = get_polygons(50, 2) +metal4_pin = metal4_pin.not(nodrc_drw) +count = metal4_pin.count() +logger.info("metal4_pin has #{count} polygons") +polygons_count += count + +metal4_mask = get_polygons(50, 20) +metal4_mask = metal4_mask.not(nodrc_drw) +count = metal4_mask.count() +logger.info("metal4_mask has #{count} polygons") +polygons_count += count + +metal4_nofill = get_polygons(50, 23) +metal4_nofill = metal4_nofill.not(nodrc_drw) +count = metal4_nofill.count() +logger.info("metal4_nofill has #{count} polygons") +polygons_count += count + +metal4_text = labels(50, 25) +metal4_text = metal4_text.not(nodrc_drw) +count = metal4_text.count() +logger.info("metal4_text has #{count} polygons") +polygons_count += count + +metal4_OPC = get_polygons(50, 26) +metal4_OPC = metal4_OPC.not(nodrc_drw) +count = metal4_OPC.count() +logger.info("metal4_OPC has #{count} polygons") +polygons_count += count + +metal4_noqrc = get_polygons(50, 28) +metal4_noqrc = metal4_noqrc.not(nodrc_drw) +count = metal4_noqrc.count() +logger.info("metal4_noqrc has #{count} polygons") +polygons_count += count + +metal4_res = get_polygons(50, 29) +metal4_res = metal4_res.not(nodrc_drw) +count = metal4_res.count() +logger.info("metal4_res has #{count} polygons") +polygons_count += count + +metal4_iprobe = get_polygons(50, 33) +metal4_iprobe = metal4_iprobe.not(nodrc_drw) +count = metal4_iprobe.count() +logger.info("metal4_iprobe has #{count} polygons") +polygons_count += count + +metal4_diffprb = get_polygons(50, 34) +metal4_diffprb = metal4_diffprb.not(nodrc_drw) +count = metal4_diffprb.count() +logger.info("metal4_diffprb has #{count} polygons") +polygons_count += count + +heattrans_drw = get_polygons(51, 0) +heattrans_drw = heattrans_drw.not(nodrc_drw) +count = heattrans_drw.count() +logger.info("heattrans_drw has #{count} polygons") +polygons_count += count + +heatres_drw = get_polygons(52, 0) +heatres_drw = heatres_drw.not(nodrc_drw) +count = heatres_drw.count() +logger.info("heatres_drw has #{count} polygons") +polygons_count += count + +fbe_drw = get_polygons(54, 0) +fbe_drw = fbe_drw.not(nodrc_drw) +count = fbe_drw.count() +logger.info("fbe_drw has #{count} polygons") +polygons_count += count + +empoly_drw = get_polygons(55, 0) +empoly_drw = empoly_drw.not(nodrc_drw) +count = empoly_drw.count() +logger.info("empoly_drw has #{count} polygons") +polygons_count += count + +digisub_drw = get_polygons(60, 0) +digisub_drw = digisub_drw.not(nodrc_drw) +count = digisub_drw.count() +logger.info("digisub_drw has #{count} polygons") +polygons_count += count + +text_drw = labels(63, 0) +text_drw = text_drw.not(nodrc_drw) +count = text_drw.count() +logger.info("text_drw has #{count} polygons") +polygons_count += count + +via4_drw = get_polygons(66, 0) +via4_drw = via4_drw.not(nodrc_drw) +count = via4_drw.count() +logger.info("via4_drw has #{count} polygons") +polygons_count += count + +metal5_drw = get_polygons(67, 0) +metal5_drw = metal5_drw.not(nodrc_drw) +count = metal5_drw.count() +logger.info("metal5_drw has #{count} polygons") +polygons_count += count + +metal5_filler = get_polygons(67, 22) +metal5_filler = metal5_filler.not(nodrc_drw) +count = metal5_filler.count() +logger.info("metal5_filler has #{count} polygons") +polygons_count += count + +metal5_slit = get_polygons(67, 24) +metal5_slit = metal5_slit.not(nodrc_drw) +count = metal5_slit.count() +logger.info("metal5_slit has #{count} polygons") +polygons_count += count + +# metal5 org +metal5 = metal5_drw.join(metal5_filler).not(metal5_slit) +count = metal5.count() +logger.info("metal5 has #{count} polygons") + +metal5_pin = get_polygons(67, 2) +metal5_pin = metal5_pin.not(nodrc_drw) +count = metal5_pin.count() +logger.info("metal5_pin has #{count} polygons") +polygons_count += count + +metal5_mask = get_polygons(67, 20) +metal5_mask = metal5_mask.not(nodrc_drw) +count = metal5_mask.count() +logger.info("metal5_mask has #{count} polygons") +polygons_count += count + +metal5_nofill = get_polygons(67, 23) +metal5_nofill = metal5_nofill.not(nodrc_drw) +count = metal5_nofill.count() +logger.info("metal5_nofill has #{count} polygons") +polygons_count += count + +metal5_text = labels(67, 25) +metal5_text = metal5_text.not(nodrc_drw) +count = metal5_text.count() +logger.info("metal5_text has #{count} polygons") +polygons_count += count + +metal5_OPC = get_polygons(67, 26) +metal5_OPC = metal5_OPC.not(nodrc_drw) +count = metal5_OPC.count() +logger.info("metal5_OPC has #{count} polygons") +polygons_count += count + +metal5_noqrc = get_polygons(67, 28) +metal5_noqrc = metal5_noqrc.not(nodrc_drw) +count = metal5_noqrc.count() +logger.info("metal5_noqrc has #{count} polygons") +polygons_count += count + +metal5_res = get_polygons(67, 29) +metal5_res = metal5_res.not(nodrc_drw) +count = metal5_res.count() +logger.info("metal5_res has #{count} polygons") +polygons_count += count + +metal5_iprobe = get_polygons(67, 33) +metal5_iprobe = metal5_iprobe.not(nodrc_drw) +count = metal5_iprobe.count() +logger.info("metal5_iprobe has #{count} polygons") +polygons_count += count + +metal5_diffprb = get_polygons(67, 34) +metal5_diffprb = metal5_diffprb.not(nodrc_drw) +count = metal5_diffprb.count() +logger.info("metal5_diffprb has #{count} polygons") +polygons_count += count + +radhard_drw = get_polygons(68, 0) +radhard_drw = radhard_drw.not(nodrc_drw) +count = radhard_drw.count() +logger.info("radhard_drw has #{count} polygons") +polygons_count += count + +memcap_drw = get_polygons(69, 0) +memcap_drw = memcap_drw.not(nodrc_drw) +count = memcap_drw.count() +logger.info("memcap_drw has #{count} polygons") +polygons_count += count + +varicap_drw = get_polygons(70, 0) +varicap_drw = varicap_drw.not(nodrc_drw) +count = varicap_drw.count() +logger.info("varicap_drw has #{count} polygons") +polygons_count += count + +intbondvia_drw = get_polygons(72, 0) +intbondvia_drw = intbondvia_drw.not(nodrc_drw) +count = intbondvia_drw.count() +logger.info("intbondvia_drw has #{count} polygons") +polygons_count += count + +intbondmet_drw = get_polygons(73, 0) +intbondmet_drw = intbondmet_drw.not(nodrc_drw) +count = intbondmet_drw.count() +logger.info("intbondmet_drw has #{count} polygons") +polygons_count += count + +devbondvia_drw = get_polygons(74, 0) +devbondvia_drw = devbondvia_drw.not(nodrc_drw) +count = devbondvia_drw.count() +logger.info("devbondvia_drw has #{count} polygons") +polygons_count += count + +devbondmet_drw = get_polygons(75, 0) +devbondmet_drw = devbondmet_drw.not(nodrc_drw) +count = devbondmet_drw.count() +logger.info("devbondmet_drw has #{count} polygons") +polygons_count += count + +devtrench_drw = get_polygons(76, 0) +devtrench_drw = devtrench_drw.not(nodrc_drw) +count = devtrench_drw.count() +logger.info("devtrench_drw has #{count} polygons") +polygons_count += count + +redist_drw = get_polygons(77, 0) +redist_drw = redist_drw.not(nodrc_drw) +count = redist_drw.count() +logger.info("redist_drw has #{count} polygons") +polygons_count += count + +graphbot_drw = get_polygons(78, 0) +graphbot_drw = graphbot_drw.not(nodrc_drw) +count = graphbot_drw.count() +logger.info("graphbot_drw has #{count} polygons") +polygons_count += count + +graphtop_drw = get_polygons(79, 0) +graphtop_drw = graphtop_drw.not(nodrc_drw) +count = graphtop_drw.count() +logger.info("graphtop_drw has #{count} polygons") +polygons_count += count + +antvia1_drw = get_polygons(83, 0) +antvia1_drw = antvia1_drw.not(nodrc_drw) +count = antvia1_drw.count() +logger.info("antvia1_drw has #{count} polygons") +polygons_count += count + +antmetal2_drw = get_polygons(84, 0) +antmetal2_drw = antmetal2_drw.not(nodrc_drw) +count = antmetal2_drw.count() +logger.info("antmetal2_drw has #{count} polygons") +polygons_count += count + +graphcont_drw = get_polygons(85, 0) +graphcont_drw = graphcont_drw.not(nodrc_drw) +count = graphcont_drw.count() +logger.info("graphcont_drw has #{count} polygons") +polygons_count += count + +siwg_drw = get_polygons(86, 0) +siwg_drw = siwg_drw.not(nodrc_drw) +count = siwg_drw.count() +logger.info("siwg_drw has #{count} polygons") +polygons_count += count + +siwg_filler = get_polygons(86, 22) +siwg_filler = siwg_filler.not(nodrc_drw) +count = siwg_filler.count() +logger.info("siwg_filler has #{count} polygons") +polygons_count += count + +siwg_nofill = get_polygons(86, 23) +siwg_nofill = siwg_nofill.not(nodrc_drw) +count = siwg_nofill.count() +logger.info("siwg_nofill has #{count} polygons") +polygons_count += count + +sigrating_drw = get_polygons(87, 0) +sigrating_drw = sigrating_drw.not(nodrc_drw) +count = sigrating_drw.count() +logger.info("sigrating_drw has #{count} polygons") +polygons_count += count + +singrating_drw = get_polygons(88, 0) +singrating_drw = singrating_drw.not(nodrc_drw) +count = singrating_drw.count() +logger.info("singrating_drw has #{count} polygons") +polygons_count += count + +graphpas_drw = get_polygons(89, 0) +graphpas_drw = graphpas_drw.not(nodrc_drw) +count = graphpas_drw.count() +logger.info("graphpas_drw has #{count} polygons") +polygons_count += count + +emwind3_drw = get_polygons(90, 0) +emwind3_drw = emwind3_drw.not(nodrc_drw) +count = emwind3_drw.count() +logger.info("emwind3_drw has #{count} polygons") +polygons_count += count + +emwihv3_drw = get_polygons(91, 0) +emwihv3_drw = emwihv3_drw.not(nodrc_drw) +count = emwihv3_drw.count() +logger.info("emwihv3_drw has #{count} polygons") +polygons_count += count + +redbulay_drw = get_polygons(92, 0) +redbulay_drw = redbulay_drw.not(nodrc_drw) +count = redbulay_drw.count() +logger.info("redbulay_drw has #{count} polygons") +polygons_count += count + +smos_drw = get_polygons(93, 0) +smos_drw = smos_drw.not(nodrc_drw) +count = smos_drw.count() +logger.info("smos_drw has #{count} polygons") +polygons_count += count + +graphpad_drw = get_polygons(97, 0) +graphpad_drw = graphpad_drw.not(nodrc_drw) +count = graphpad_drw.count() +logger.info("graphpad_drw has #{count} polygons") +polygons_count += count + +polimide_drw = get_polygons(98, 0) +polimide_drw = polimide_drw.not(nodrc_drw) +count = polimide_drw.count() +logger.info("polimide_drw has #{count} polygons") +polygons_count += count + +polimide_pin = get_polygons(98, 2) +polimide_pin = polimide_pin.not(nodrc_drw) +count = polimide_pin.count() +logger.info("polimide_pin has #{count} polygons") +polygons_count += count + +recog_drw = get_polygons(99, 0) +recog_drw = recog_drw.not(nodrc_drw) +count = recog_drw.count() +logger.info("recog_drw has #{count} polygons") +polygons_count += count + +recog_pin = get_polygons(99, 2) +recog_pin = recog_pin.not(nodrc_drw) +count = recog_pin.count() +logger.info("recog_pin has #{count} polygons") +polygons_count += count + +recog_esd = get_polygons(99, 30) +recog_esd = recog_esd.not(nodrc_drw) +count = recog_esd.count() +logger.info("recog_esd has #{count} polygons") +polygons_count += count + +recog_diode = get_polygons(99, 31) +recog_diode = recog_diode.not(nodrc_drw) +count = recog_diode.count() +logger.info("recog_diode has #{count} polygons") +polygons_count += count + +recog_tsv = get_polygons(99, 32) +recog_tsv = recog_tsv.not(nodrc_drw) +count = recog_tsv.count() +logger.info("recog_tsv has #{count} polygons") +polygons_count += count + +recog_iprobe = get_polygons(99, 33) +recog_iprobe = recog_iprobe.not(nodrc_drw) +count = recog_iprobe.count() +logger.info("recog_iprobe has #{count} polygons") +polygons_count += count + +recog_diffprb = get_polygons(99, 34) +recog_diffprb = recog_diffprb.not(nodrc_drw) +count = recog_diffprb.count() +logger.info("recog_diffprb has #{count} polygons") +polygons_count += count + +recog_pillar = get_polygons(99, 35) +recog_pillar = recog_pillar.not(nodrc_drw) +count = recog_pillar.count() +logger.info("recog_pillar has #{count} polygons") +polygons_count += count + +recog_sbump = get_polygons(99, 36) +recog_sbump = recog_sbump.not(nodrc_drw) +count = recog_sbump.count() +logger.info("recog_sbump has #{count} polygons") +polygons_count += count + +recog_otp = get_polygons(99, 37) +recog_otp = recog_otp.not(nodrc_drw) +count = recog_otp.count() +logger.info("recog_otp has #{count} polygons") +polygons_count += count + +recog_pdiode = get_polygons(99, 38) +recog_pdiode = recog_pdiode.not(nodrc_drw) +count = recog_pdiode.count() +logger.info("recog_pdiode has #{count} polygons") +polygons_count += count + +recog_mom = get_polygons(99, 39) +recog_mom = recog_mom.not(nodrc_drw) +count = recog_mom.count() +logger.info("recog_mom has #{count} polygons") +polygons_count += count + +recog_pcm = get_polygons(99, 100) +recog_pcm = recog_pcm.not(nodrc_drw) +count = recog_pcm.count() +logger.info("recog_pcm has #{count} polygons") +polygons_count += count + +colopen_drw = get_polygons(101, 0) +colopen_drw = colopen_drw.not(nodrc_drw) +count = colopen_drw.count() +logger.info("colopen_drw has #{count} polygons") +polygons_count += count + +graphmetal1_drw = get_polygons(109, 0) +graphmetal1_drw = graphmetal1_drw.not(nodrc_drw) +count = graphmetal1_drw.count() +logger.info("graphmetal1_drw has #{count} polygons") +polygons_count += count + +graphmetal1_filler = get_polygons(109, 22) +graphmetal1_filler = graphmetal1_filler.not(nodrc_drw) +count = graphmetal1_filler.count() +logger.info("graphmetal1_filler has #{count} polygons") +polygons_count += count + +graphmetal1_nofill = get_polygons(109, 23) +graphmetal1_nofill = graphmetal1_nofill.not(nodrc_drw) +count = graphmetal1_nofill.count() +logger.info("graphmetal1_nofill has #{count} polygons") +polygons_count += count + +graphmetal1_slit = get_polygons(109, 24) +graphmetal1_slit = graphmetal1_slit.not(nodrc_drw) +count = graphmetal1_slit.count() +logger.info("graphmetal1_slit has #{count} polygons") +polygons_count += count + +graphmetal1_OPC = get_polygons(109, 26) +graphmetal1_OPC = graphmetal1_OPC.not(nodrc_drw) +count = graphmetal1_OPC.count() +logger.info("graphmetal1_OPC has #{count} polygons") +polygons_count += count + +graphmet1l_drw = get_polygons(110, 0) +graphmet1l_drw = graphmet1l_drw.not(nodrc_drw) +count = graphmet1l_drw.count() +logger.info("graphmet1l_drw has #{count} polygons") +polygons_count += count + +graphmet1l_filler = get_polygons(110, 22) +graphmet1l_filler = graphmet1l_filler.not(nodrc_drw) +count = graphmet1l_filler.count() +logger.info("graphmet1l_filler has #{count} polygons") +polygons_count += count + +graphmet1l_nofill = get_polygons(110, 23) +graphmet1l_nofill = graphmet1l_nofill.not(nodrc_drw) +count = graphmet1l_nofill.count() +logger.info("graphmet1l_nofill has #{count} polygons") +polygons_count += count + +graphmet1l_slit = get_polygons(110, 24) +graphmet1l_slit = graphmet1l_slit.not(nodrc_drw) +count = graphmet1l_slit.count() +logger.info("graphmet1l_slit has #{count} polygons") +polygons_count += count + +graphmet1l_OPC = get_polygons(110, 26) +graphmet1l_OPC = graphmet1l_OPC.not(nodrc_drw) +count = graphmet1l_OPC.count() +logger.info("graphmet1l_OPC has #{count} polygons") +polygons_count += count + +extblock_drw = get_polygons(111, 0) +extblock_drw = extblock_drw.not(nodrc_drw) +count = extblock_drw.count() +logger.info("extblock_drw has #{count} polygons") +polygons_count += count + +nldd_drw = get_polygons(112, 0) +nldd_drw = nldd_drw.not(nodrc_drw) +count = nldd_drw.count() +logger.info("nldd_drw has #{count} polygons") +polygons_count += count + +pldd_drw = get_polygons(113, 0) +pldd_drw = pldd_drw.not(nodrc_drw) +count = pldd_drw.count() +logger.info("pldd_drw has #{count} polygons") +polygons_count += count + +next_drw = get_polygons(114, 0) +next_drw = next_drw.not(nodrc_drw) +count = next_drw.count() +logger.info("next_drw has #{count} polygons") +polygons_count += count + +pext_drw = get_polygons(115, 0) +pext_drw = pext_drw.not(nodrc_drw) +count = pext_drw.count() +logger.info("pext_drw has #{count} polygons") +polygons_count += count + +nexthv_drw = get_polygons(116, 0) +nexthv_drw = nexthv_drw.not(nodrc_drw) +count = nexthv_drw.count() +logger.info("nexthv_drw has #{count} polygons") +polygons_count += count + +pexthv_drw = get_polygons(117, 0) +pexthv_drw = pexthv_drw.not(nodrc_drw) +count = pexthv_drw.count() +logger.info("pexthv_drw has #{count} polygons") +polygons_count += count + +graphgate_drw = get_polygons(118, 0) +graphgate_drw = graphgate_drw.not(nodrc_drw) +count = graphgate_drw.count() +logger.info("graphgate_drw has #{count} polygons") +polygons_count += count + +sinwg_drw = get_polygons(119, 0) +sinwg_drw = sinwg_drw.not(nodrc_drw) +count = sinwg_drw.count() +logger.info("sinwg_drw has #{count} polygons") +polygons_count += count + +sinwg_filler = get_polygons(119, 22) +sinwg_filler = sinwg_filler.not(nodrc_drw) +count = sinwg_filler.count() +logger.info("sinwg_filler has #{count} polygons") +polygons_count += count + +sinwg_nofill = get_polygons(119, 23) +sinwg_nofill = sinwg_nofill.not(nodrc_drw) +count = sinwg_nofill.count() +logger.info("sinwg_nofill has #{count} polygons") +polygons_count += count + +mempad_drw = get_polygons(124, 0) +mempad_drw = mempad_drw.not(nodrc_drw) +count = mempad_drw.count() +logger.info("mempad_drw has #{count} polygons") +polygons_count += count + +topvia1_drw = get_polygons(125, 0) +topvia1_drw = topvia1_drw.not(nodrc_drw) +count = topvia1_drw.count() +logger.info("topvia1_drw has #{count} polygons") +polygons_count += count + +topmetal1_drw = get_polygons(126, 0) +topmetal1_drw = topmetal1_drw.not(nodrc_drw) +count = topmetal1_drw.count() +logger.info("topmetal1_drw has #{count} polygons") +polygons_count += count + +topmetal1_filler = get_polygons(126, 22) +topmetal1_filler = topmetal1_filler.not(nodrc_drw) +count = topmetal1_filler.count() +logger.info("topmetal1_filler has #{count} polygons") +polygons_count += count + +topmetal1_slit = get_polygons(126, 24) +topmetal1_slit = topmetal1_slit.not(nodrc_drw) +count = topmetal1_slit.count() +logger.info("topmetal1_slit has #{count} polygons") +polygons_count += count + +# topmetal1 org +topmetal1 = topmetal1_drw.join(topmetal1_filler).not(topmetal1_slit) +count = topmetal1.count() +logger.info("topmetal1 has #{count} polygons") + +topmetal1_pin = get_polygons(126, 2) +topmetal1_pin = topmetal1_pin.not(nodrc_drw) +count = topmetal1_pin.count() +logger.info("topmetal1_pin has #{count} polygons") +polygons_count += count + +topmetal1_mask = get_polygons(126, 20) +topmetal1_mask = topmetal1_mask.not(nodrc_drw) +count = topmetal1_mask.count() +logger.info("topmetal1_mask has #{count} polygons") +polygons_count += count + +topmetal1_nofill = get_polygons(126, 23) +topmetal1_nofill = topmetal1_nofill.not(nodrc_drw) +count = topmetal1_nofill.count() +logger.info("topmetal1_nofill has #{count} polygons") +polygons_count += count + +topmetal1_text = labels(126, 25) +topmetal1_text = topmetal1_text.not(nodrc_drw) +count = topmetal1_text.count() +logger.info("topmetal1_text has #{count} polygons") +polygons_count += count + +topmetal1_noqrc = get_polygons(126, 28) +topmetal1_noqrc = topmetal1_noqrc.not(nodrc_drw) +count = topmetal1_noqrc.count() +logger.info("topmetal1_noqrc has #{count} polygons") +polygons_count += count + +topmetal1_res = get_polygons(126, 29) +topmetal1_res = topmetal1_res.not(nodrc_drw) +count = topmetal1_res.count() +logger.info("topmetal1_res has #{count} polygons") +polygons_count += count + +topmetal1_iprobe = get_polygons(126, 33) +topmetal1_iprobe = topmetal1_iprobe.not(nodrc_drw) +count = topmetal1_iprobe.count() +logger.info("topmetal1_iprobe has #{count} polygons") +polygons_count += count + +topmetal1_diffprb = get_polygons(126, 34) +topmetal1_diffprb = topmetal1_diffprb.not(nodrc_drw) +count = topmetal1_diffprb.count() +logger.info("topmetal1_diffprb has #{count} polygons") +polygons_count += count + +inldpwl_drw = get_polygons(127, 0) +inldpwl_drw = inldpwl_drw.not(nodrc_drw) +count = inldpwl_drw.count() +logger.info("inldpwl_drw has #{count} polygons") +polygons_count += count + +polyres_drw = get_polygons(128, 0) +polyres_drw = polyres_drw.not(nodrc_drw) +count = polyres_drw.count() +logger.info("polyres_drw has #{count} polygons") +polygons_count += count + +polyres_pin = get_polygons(128, 2) +polyres_pin = polyres_pin.not(nodrc_drw) +count = polyres_pin.count() +logger.info("polyres_pin has #{count} polygons") +polygons_count += count + +vmim_drw = get_polygons(129, 0) +vmim_drw = vmim_drw.not(nodrc_drw) +count = vmim_drw.count() +logger.info("vmim_drw has #{count} polygons") +polygons_count += count + +nbulaycut_drw = get_polygons(131, 0) +nbulaycut_drw = nbulaycut_drw.not(nodrc_drw) +count = nbulaycut_drw.count() +logger.info("nbulaycut_drw has #{count} polygons") +polygons_count += count + +antmetal1_drw = get_polygons(132, 0) +antmetal1_drw = antmetal1_drw.not(nodrc_drw) +count = antmetal1_drw.count() +logger.info("antmetal1_drw has #{count} polygons") +polygons_count += count + +topvia2_drw = get_polygons(133, 0) +topvia2_drw = topvia2_drw.not(nodrc_drw) +count = topvia2_drw.count() +logger.info("topvia2_drw has #{count} polygons") +polygons_count += count + +topmetal2_drw = get_polygons(134, 0) +topmetal2_drw = topmetal2_drw.not(nodrc_drw) +count = topmetal2_drw.count() +logger.info("topmetal2_drw has #{count} polygons") +polygons_count += count + +topmetal2_filler = get_polygons(134, 22) +topmetal2_filler = topmetal2_filler.not(nodrc_drw) +count = topmetal2_filler.count() +logger.info("topmetal2_filler has #{count} polygons") +polygons_count += count + +topmetal2_slit = get_polygons(134, 24) +topmetal2_slit = topmetal2_slit.not(nodrc_drw) +count = topmetal2_slit.count() +logger.info("topmetal2_slit has #{count} polygons") +polygons_count += count + +# topmetal2 org +topmetal2 = topmetal2_drw.join(topmetal2_filler).not(topmetal2_slit) +count = topmetal2.count() +logger.info("topmetal2 has #{count} polygons") + +topmetal2_pin = get_polygons(134, 2) +topmetal2_pin = topmetal2_pin.not(nodrc_drw) +count = topmetal2_pin.count() +logger.info("topmetal2_pin has #{count} polygons") +polygons_count += count + +topmetal2_mask = get_polygons(134, 20) +topmetal2_mask = topmetal2_mask.not(nodrc_drw) +count = topmetal2_mask.count() +logger.info("topmetal2_mask has #{count} polygons") +polygons_count += count + +topmetal2_nofill = get_polygons(134, 23) +topmetal2_nofill = topmetal2_nofill.not(nodrc_drw) +count = topmetal2_nofill.count() +logger.info("topmetal2_nofill has #{count} polygons") +polygons_count += count + +topmetal2_text = labels(134, 25) +topmetal2_text = topmetal2_text.not(nodrc_drw) +count = topmetal2_text.count() +logger.info("topmetal2_text has #{count} polygons") +polygons_count += count + +topmetal2_noqrc = get_polygons(134, 28) +topmetal2_noqrc = topmetal2_noqrc.not(nodrc_drw) +count = topmetal2_noqrc.count() +logger.info("topmetal2_noqrc has #{count} polygons") +polygons_count += count + +topmetal2_res = get_polygons(134, 29) +topmetal2_res = topmetal2_res.not(nodrc_drw) +count = topmetal2_res.count() +logger.info("topmetal2_res has #{count} polygons") +polygons_count += count + +topmetal2_iprobe = get_polygons(134, 33) +topmetal2_iprobe = topmetal2_iprobe.not(nodrc_drw) +count = topmetal2_iprobe.count() +logger.info("topmetal2_iprobe has #{count} polygons") +polygons_count += count + +topmetal2_diffprb = get_polygons(134, 34) +topmetal2_diffprb = topmetal2_diffprb.not(nodrc_drw) +count = topmetal2_diffprb.count() +logger.info("topmetal2_diffprb has #{count} polygons") +polygons_count += count + +snsring_drw = get_polygons(135, 0) +snsring_drw = snsring_drw.not(nodrc_drw) +count = snsring_drw.count() +logger.info("snsring_drw has #{count} polygons") +polygons_count += count + +sensor_drw = get_polygons(136, 0) +sensor_drw = sensor_drw.not(nodrc_drw) +count = sensor_drw.count() +logger.info("sensor_drw has #{count} polygons") +polygons_count += count + +snsarms_drw = get_polygons(137, 0) +snsarms_drw = snsarms_drw.not(nodrc_drw) +count = snsarms_drw.count() +logger.info("snsarms_drw has #{count} polygons") +polygons_count += count + +snscmosvia_drw = get_polygons(138, 0) +snscmosvia_drw = snscmosvia_drw.not(nodrc_drw) +count = snscmosvia_drw.count() +logger.info("snscmosvia_drw has #{count} polygons") +polygons_count += count + +colwind_drw = get_polygons(139, 0) +colwind_drw = colwind_drw.not(nodrc_drw) +count = colwind_drw.count() +logger.info("colwind_drw has #{count} polygons") +polygons_count += count + +flm_drw = get_polygons(142, 0) +flm_drw = flm_drw.not(nodrc_drw) +count = flm_drw.count() +logger.info("flm_drw has #{count} polygons") +polygons_count += count + +hafniumox_drw = get_polygons(143, 0) +hafniumox_drw = hafniumox_drw.not(nodrc_drw) +count = hafniumox_drw.count() +logger.info("hafniumox_drw has #{count} polygons") +polygons_count += count + +memvia_drw = get_polygons(145, 0) +memvia_drw = memvia_drw.not(nodrc_drw) +count = memvia_drw.count() +logger.info("memvia_drw has #{count} polygons") +polygons_count += count + +thinfilmres_drw = get_polygons(146, 0) +thinfilmres_drw = thinfilmres_drw.not(nodrc_drw) +count = thinfilmres_drw.count() +logger.info("thinfilmres_drw has #{count} polygons") +polygons_count += count + +rfmem_drw = get_polygons(147, 0) +rfmem_drw = rfmem_drw.not(nodrc_drw) +count = rfmem_drw.count() +logger.info("rfmem_drw has #{count} polygons") +polygons_count += count + +norcx_drw = get_polygons(148, 0) +norcx_drw = norcx_drw.not(nodrc_drw) +count = norcx_drw.count() +logger.info("norcx_drw has #{count} polygons") +polygons_count += count + +norcx_m2m3 = get_polygons(148, 41) +norcx_m2m3 = norcx_m2m3.not(nodrc_drw) +count = norcx_m2m3.count() +logger.info("norcx_m2m3 has #{count} polygons") +polygons_count += count + +norcx_m2m4 = get_polygons(148, 42) +norcx_m2m4 = norcx_m2m4.not(nodrc_drw) +count = norcx_m2m4.count() +logger.info("norcx_m2m4 has #{count} polygons") +polygons_count += count + +norcx_m2m5 = get_polygons(148, 43) +norcx_m2m5 = norcx_m2m5.not(nodrc_drw) +count = norcx_m2m5.count() +logger.info("norcx_m2m5 has #{count} polygons") +polygons_count += count + +norcx_m2tm1 = get_polygons(148, 44) +norcx_m2tm1 = norcx_m2tm1.not(nodrc_drw) +count = norcx_m2tm1.count() +logger.info("norcx_m2tm1 has #{count} polygons") +polygons_count += count + +norcx_m2tm2 = get_polygons(148, 45) +norcx_m2tm2 = norcx_m2tm2.not(nodrc_drw) +count = norcx_m2tm2.count() +logger.info("norcx_m2tm2 has #{count} polygons") +polygons_count += count + +norcx_m3m4 = get_polygons(148, 46) +norcx_m3m4 = norcx_m3m4.not(nodrc_drw) +count = norcx_m3m4.count() +logger.info("norcx_m3m4 has #{count} polygons") +polygons_count += count + +norcx_m3m5 = get_polygons(148, 47) +norcx_m3m5 = norcx_m3m5.not(nodrc_drw) +count = norcx_m3m5.count() +logger.info("norcx_m3m5 has #{count} polygons") +polygons_count += count + +norcx_m3tm1 = get_polygons(148, 48) +norcx_m3tm1 = norcx_m3tm1.not(nodrc_drw) +count = norcx_m3tm1.count() +logger.info("norcx_m3tm1 has #{count} polygons") +polygons_count += count + +norcx_m3tm2 = get_polygons(148, 49) +norcx_m3tm2 = norcx_m3tm2.not(nodrc_drw) +count = norcx_m3tm2.count() +logger.info("norcx_m3tm2 has #{count} polygons") +polygons_count += count + +norcx_m4m5 = get_polygons(148, 50) +norcx_m4m5 = norcx_m4m5.not(nodrc_drw) +count = norcx_m4m5.count() +logger.info("norcx_m4m5 has #{count} polygons") +polygons_count += count + +norcx_m4tm1 = get_polygons(148, 51) +norcx_m4tm1 = norcx_m4tm1.not(nodrc_drw) +count = norcx_m4tm1.count() +logger.info("norcx_m4tm1 has #{count} polygons") +polygons_count += count + +norcx_m4tm2 = get_polygons(148, 52) +norcx_m4tm2 = norcx_m4tm2.not(nodrc_drw) +count = norcx_m4tm2.count() +logger.info("norcx_m4tm2 has #{count} polygons") +polygons_count += count + +norcx_m5tm1 = get_polygons(148, 53) +norcx_m5tm1 = norcx_m5tm1.not(nodrc_drw) +count = norcx_m5tm1.count() +logger.info("norcx_m5tm1 has #{count} polygons") +polygons_count += count + +norcx_m5tm2 = get_polygons(148, 54) +norcx_m5tm2 = norcx_m5tm2.not(nodrc_drw) +count = norcx_m5tm2.count() +logger.info("norcx_m5tm2 has #{count} polygons") +polygons_count += count + +norcx_tm1tm2 = get_polygons(148, 55) +norcx_tm1tm2 = norcx_tm1tm2.not(nodrc_drw) +count = norcx_tm1tm2.count() +logger.info("norcx_tm1tm2 has #{count} polygons") +polygons_count += count + +norcx_m1sub = get_polygons(148, 123) +norcx_m1sub = norcx_m1sub.not(nodrc_drw) +count = norcx_m1sub.count() +logger.info("norcx_m1sub has #{count} polygons") +polygons_count += count + +norcx_m2sub = get_polygons(148, 124) +norcx_m2sub = norcx_m2sub.not(nodrc_drw) +count = norcx_m2sub.count() +logger.info("norcx_m2sub has #{count} polygons") +polygons_count += count + +norcx_m3sub = get_polygons(148, 125) +norcx_m3sub = norcx_m3sub.not(nodrc_drw) +count = norcx_m3sub.count() +logger.info("norcx_m3sub has #{count} polygons") +polygons_count += count + +norcx_m4sub = get_polygons(148, 126) +norcx_m4sub = norcx_m4sub.not(nodrc_drw) +count = norcx_m4sub.count() +logger.info("norcx_m4sub has #{count} polygons") +polygons_count += count + +norcx_m5sub = get_polygons(148, 127) +norcx_m5sub = norcx_m5sub.not(nodrc_drw) +count = norcx_m5sub.count() +logger.info("norcx_m5sub has #{count} polygons") +polygons_count += count + +norcx_tm1sub = get_polygons(148, 300) +norcx_tm1sub = norcx_tm1sub.not(nodrc_drw) +count = norcx_tm1sub.count() +logger.info("norcx_tm1sub has #{count} polygons") +polygons_count += count + +norcx_tm2sub = get_polygons(148, 301) +norcx_tm2sub = norcx_tm2sub.not(nodrc_drw) +count = norcx_tm2sub.count() +logger.info("norcx_tm2sub has #{count} polygons") +polygons_count += count + +snsbotvia_drw = get_polygons(149, 0) +snsbotvia_drw = snsbotvia_drw.not(nodrc_drw) +count = snsbotvia_drw.count() +logger.info("snsbotvia_drw has #{count} polygons") +polygons_count += count + +snstopvia_drw = get_polygons(151, 0) +snstopvia_drw = snstopvia_drw.not(nodrc_drw) +count = snstopvia_drw.count() +logger.info("snstopvia_drw has #{count} polygons") +polygons_count += count + +deepvia_drw = get_polygons(152, 0) +deepvia_drw = deepvia_drw.not(nodrc_drw) +count = deepvia_drw.count() +logger.info("deepvia_drw has #{count} polygons") +polygons_count += count + +fgetch_drw = get_polygons(153, 0) +fgetch_drw = fgetch_drw.not(nodrc_drw) +count = fgetch_drw.count() +logger.info("fgetch_drw has #{count} polygons") +polygons_count += count + +ctrgat_drw = get_polygons(154, 0) +ctrgat_drw = ctrgat_drw.not(nodrc_drw) +count = ctrgat_drw.count() +logger.info("ctrgat_drw has #{count} polygons") +polygons_count += count + +fgimp_drw = get_polygons(155, 0) +fgimp_drw = fgimp_drw.not(nodrc_drw) +count = fgimp_drw.count() +logger.info("fgimp_drw has #{count} polygons") +polygons_count += count + +emwihv_drw = get_polygons(156, 0) +emwihv_drw = emwihv_drw.not(nodrc_drw) +count = emwihv_drw.count() +logger.info("emwihv_drw has #{count} polygons") +polygons_count += count + +lbe_drw = get_polygons(157, 0) +lbe_drw = lbe_drw.not(nodrc_drw) +count = lbe_drw.count() +logger.info("lbe_drw has #{count} polygons") +polygons_count += count + +alcustop_drw = get_polygons(159, 0) +alcustop_drw = alcustop_drw.not(nodrc_drw) +count = alcustop_drw.count() +logger.info("alcustop_drw has #{count} polygons") +polygons_count += count + +nometfiller_drw = get_polygons(160, 0) +nometfiller_drw = nometfiller_drw.not(nodrc_drw) +count = nometfiller_drw.count() +logger.info("nometfiller_drw has #{count} polygons") +polygons_count += count + +prboundary_drw = get_polygons(189, 0) +prboundary_drw = prboundary_drw.not(nodrc_drw) +count = prboundary_drw.count() +logger.info("prboundary_drw has #{count} polygons") +polygons_count += count + +exchange0_drw = get_polygons(190, 0) +exchange0_drw = exchange0_drw.not(nodrc_drw) +count = exchange0_drw.count() +logger.info("exchange0_drw has #{count} polygons") +polygons_count += count + +exchange0_pin = get_polygons(190, 2) +exchange0_pin = exchange0_pin.not(nodrc_drw) +count = exchange0_pin.count() +logger.info("exchange0_pin has #{count} polygons") +polygons_count += count + +exchange0_text = labels(190, 25) +exchange0_text = exchange0_text.not(nodrc_drw) +count = exchange0_text.count() +logger.info("exchange0_text has #{count} polygons") +polygons_count += count + +exchange1_drw = get_polygons(191, 0) +exchange1_drw = exchange1_drw.not(nodrc_drw) +count = exchange1_drw.count() +logger.info("exchange1_drw has #{count} polygons") +polygons_count += count + +exchange1_pin = get_polygons(191, 2) +exchange1_pin = exchange1_pin.not(nodrc_drw) +count = exchange1_pin.count() +logger.info("exchange1_pin has #{count} polygons") +polygons_count += count + +exchange1_text = labels(191, 25) +exchange1_text = exchange1_text.not(nodrc_drw) +count = exchange1_text.count() +logger.info("exchange1_text has #{count} polygons") +polygons_count += count + +exchange2_drw = get_polygons(192, 0) +exchange2_drw = exchange2_drw.not(nodrc_drw) +count = exchange2_drw.count() +logger.info("exchange2_drw has #{count} polygons") +polygons_count += count + +exchange2_pin = get_polygons(192, 2) +exchange2_pin = exchange2_pin.not(nodrc_drw) +count = exchange2_pin.count() +logger.info("exchange2_pin has #{count} polygons") +polygons_count += count + +exchange2_text = labels(192, 25) +exchange2_text = exchange2_text.not(nodrc_drw) +count = exchange2_text.count() +logger.info("exchange2_text has #{count} polygons") +polygons_count += count + +exchange3_drw = get_polygons(193, 0) +exchange3_drw = exchange3_drw.not(nodrc_drw) +count = exchange3_drw.count() +logger.info("exchange3_drw has #{count} polygons") +polygons_count += count + +exchange3_pin = get_polygons(193, 2) +exchange3_pin = exchange3_pin.not(nodrc_drw) +count = exchange3_pin.count() +logger.info("exchange3_pin has #{count} polygons") +polygons_count += count + +exchange3_text = labels(193, 25) +exchange3_text = exchange3_text.not(nodrc_drw) +count = exchange3_text.count() +logger.info("exchange3_text has #{count} polygons") +polygons_count += count + +exchange4_drw = get_polygons(194, 0) +exchange4_drw = exchange4_drw.not(nodrc_drw) +count = exchange4_drw.count() +logger.info("exchange4_drw has #{count} polygons") +polygons_count += count + +exchange4_pin = get_polygons(194, 2) +exchange4_pin = exchange4_pin.not(nodrc_drw) +count = exchange4_pin.count() +logger.info("exchange4_pin has #{count} polygons") +polygons_count += count + +exchange4_text = labels(194, 25) +exchange4_text = exchange4_text.not(nodrc_drw) +count = exchange4_text.count() +logger.info("exchange4_text has #{count} polygons") +polygons_count += count + +isonwell_drw = get_polygons(257, 0) +isonwell_drw = isonwell_drw.not(nodrc_drw) +count = isonwell_drw.count() +logger.info("isonwell_drw has #{count} polygons") +polygons_count += count + + +logger.info("Total no. of polygons in the design is #{polygons_count}") diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_connections.lvs new file mode 100644 index 00000000..ca676a72 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_connections.lvs @@ -0,0 +1,24 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ----- MOSFET CONNECTIONS ------ +#================================ + +logger.info('Starting LVS MOSFET CONNECTIONS') diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_derivations.lvs new file mode 100644 index 00000000..9dead314 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_derivations.lvs @@ -0,0 +1,83 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================== +# ------ MOSFET DERIVATIONS ------- +#================================== + +logger.info('Starting MOSFET DERIVATIONS') + +mos_exclude = pwell_block.join(nsd_drw).join(trans_drw) + .join(emwind_drw).join(emwihv_drw).join(salblock_drw) + .join(polyres_drw).join(extblock_drw).join(res_drw) + .join(activ_mask).join(recog_diode).join(recog_esd) + .join(ind_drw).join(ind_pin).join(ind_drw) + .join(substrate_drw).join(nsd_block) + +# ==== General FETs & RF-FETs ===== + +rfnmos_exc = nwell_drw.join(psd_drw).join(mos_exclude) +rfpmos_exc = pwell.join(nwell_holes).join(mos_exclude) + +# Get case insensitive patterns for FETs +rfnmos_patt = glob_to_case_insensitive_glob("rfnmos") +rfnmoshv_patt = glob_to_case_insensitive_glob("rfnmosHV") + +rfpmos_patt = glob_to_case_insensitive_glob("rfpmos") +rfpmoshv_patt = glob_to_case_insensitive_glob("rfpmosHV") + +rfnmos_mk_gen = ptap.join(ptap_holes) +rfnmos_mk = rfnmos_mk_gen.interacting(text_drw.texts(rfnmos_patt)) +rfnmoshv_mk = rfnmos_mk_gen.interacting(text_drw.texts(rfnmoshv_patt)) + +rfpmos_mk_gen = ntap.join(ntap_holes) +rfpmos_mk = rfpmos_mk_gen.interacting(text_drw.texts(rfpmos_patt)) +rfpmoshv_mk = rfpmos_mk_gen.interacting(text_drw.texts(rfpmoshv_patt)) + +# ============== +# ---- NMOS ---- +# ============== + +logger.info('Starting NMOS DERIVATIONS') + +# for regular FETs +nmos_exc = rfnmos_exc.join(rfnmos_mk) +nmoshv_exc = rfnmos_exc.join(rfnmoshv_mk) + +# nmos - LV +ngate_lv = ngate_lv_base.not(nmos_exc) + +# nmos - HV +ngate_hv = ngate_hv_base.not(nmoshv_exc) + +# ============== +# ---- PMOS ---- +# ============== + +logger.info('Starting PMOS DERIVATIONS') + +# for regular FETs +pmos_exc = rfpmos_exc.join(rfpmos_mk) +pmoshv_exc = rfpmos_exc.join(rfpmoshv_mk) + +# pmos - LV +pgate_lv = pgate_lv_base.not(pmos_exc) + +# pmos - HV +pgate_hv = pgate_hv_base.not(pmoshv_exc) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_extraction.lvs new file mode 100644 index 00000000..a6275d10 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/mos_extraction.lvs @@ -0,0 +1,77 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +# =============================== +# ------ MOSFET EXTRACTION ------ +# =============================== + +logger.info('Starting MOSFET EXTRACTION') + +# ============== +# ---- NMOS ---- +# ============== + + +logger.info('Starting NMOS EXTRACTION') + +# nmos - LV +logger.info('Extraction of NMOS-LV transistor') +extract_devices(mos4('sg13_lv_nmos'), + { 'SD' => nsd_fet, + 'G' => ngate_lv, + 'tS' => nsd_fet, + 'tD' => nsd_fet, + 'tG' => poly_con, + 'W' => pwell }) + +# nmos - HV +logger.info('Extraction of NMOS-HV transistor') +extract_devices(mos4('sg13_hv_nmos'), + { 'SD' => nsd_fet, + 'G' => ngate_hv, + 'tS' => nsd_fet, + 'tD' => nsd_fet, + 'tG' => poly_con, + 'W' => pwell }) + +# ============== +# ---- PMOS ---- +# ============== + +logger.info('Starting PMOS EXTRACTION') + +# pmos - LV +logger.info('Extraction of PMOS-LV transistor') +extract_devices(mos4('sg13_lv_pmos'), + { 'SD' => psd_fet, + 'G' => pgate_lv, + 'tS' => psd_fet, + 'tD' => psd_fet, + 'tG' => poly_con, + 'W' => nwell_drw }) + +# pmos - HV +logger.info('Extraction of PMOS-HV transistor') +extract_devices(mos4('sg13_hv_pmos'), + { 'SD' => psd_fet, + 'G' => pgate_hv, + 'tS' => psd_fet, + 'tD' => psd_fet, + 'tG' => poly_con, + 'W' => nwell_drw }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_connections.lvs new file mode 100644 index 00000000..c3ec332e --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_connections.lvs @@ -0,0 +1,36 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- RESISTOR CONNECTIONS ----- +#================================ + +logger.info('Starting RESISTOR CONNECTIONS') + +# === rsil === +connect(rsil_ports, cont_drw) + +# === rppd === +connect(rppd_ports, cont_drw) + +# === rhigh === +connect(rhigh_ports, cont_drw) + +# === Metal Res === +# Added in general connections diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_derivations.lvs new file mode 100644 index 00000000..cb85635f --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_derivations.lvs @@ -0,0 +1,75 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- RESISTOR DERIVATIONS ----- +#================================ + +logger.info('Starting RESISTOR DERIVATIONS') + +polyres_exclude = activ.join(pwell_block).join(nsd_block) + .join(nbulay_drw).join(thickgateox_drw).join(trans_drw) + .join(emwind_drw).join(emwihv_drw).join(activ_mask) + .join(recog_diode).join(recog_esd).join(ind_drw) + .join(ind_pin).join(substrate_drw) + +# ============== +# ---- POLY ---- +# ============== + +## polyres +polyres_mk = polyres_drw.and(extblock_drw).interacting(gatpoly).not(polyres_exclude) + +## rhigh +rhigh_res = polyres_mk.and(psd_drw).and(nsd_drw).and(salblock_drw) +rhigh_ports = gatpoly.interacting(rhigh_res).not(rhigh_res) + +## rppd +rppd_res = polyres_mk.and(psd_drw).and(salblock_drw).not(nsd_block).not(nsd_drw) +rppd_ports = gatpoly.interacting(rppd_res).not(rppd_res) + +## rsil +rsil_exc = psd_drw.join(salblock_drw).join(nsd_drw).join(nsd_block) +rsil_res = polyres_mk.and(res_drw).not(rsil_exc) +rsil_ports = gatpoly.interacting(rsil_res).not(rsil_res) + +# =============== +# ---- METAL ---- +# =============== + +# res_metal1 +res_metal1 = metal1.and(metal1_res) + +# res_metal2 +res_metal2 = metal2.and(metal2_res) + +# res_metal3 +res_metal3 = metal3.and(metal3_res) + +# res_metal4 +res_metal4 = metal4.and(metal4_res) + +# res_metal5 +res_metal5 = metal5.and(metal5_res) + +# res_topmetal1 +res_topmetal1 = topmetal1.and(topmetal1_res) + +# res_topmetal2 +res_topmetal2 = topmetal2.and(topmetal2_res) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_extraction.lvs new file mode 100644 index 00000000..d0483911 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/res_extraction.lvs @@ -0,0 +1,89 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- RESISTOR EXTRACTIONS ----- +#================================ + +logger.info('Starting RESISTOR EXTRACTION') + +# ============== +# ---- POLY ---- +# ============== + +# rsil +logger.info('Extracting rsil resistor') +extract_devices(GeneralNTerminalExtractor.new('rsil', 2), { + 'core' => rsil_res, + 'ports' => rsil_ports, + 'meas_mk' => polyres_drw, + 'dev_mk' => polyres_drw.interacting(rsil_res), + 'sub_mk' => pwell.join(nwell_drw) + }) +# rppd +logger.info('Extracting rppd resistor') +extract_devices(GeneralNTerminalExtractor.new('rppd', 2), { + 'core' => rppd_res, + 'ports' => rppd_ports, + 'meas_mk' => polyres_drw, + 'dev_mk' => polyres_drw.interacting(rppd_res), + 'sub_mk' => pwell.join(nwell_drw) + }) + +# rhigh +logger.info('Extracting rhigh resistor') +extract_devices(GeneralNTerminalExtractor.new('rhigh', 2), { + 'core' => rhigh_res, + 'ports' => rhigh_ports, + 'meas_mk' => polyres_drw, + 'dev_mk' => polyres_drw.interacting(rhigh_res), + 'sub_mk' => pwell.join(nwell_drw) + }) + +# =============== +# ---- METAL ---- +# =============== + +# RM1 +logger.info('Extracting res_metal1 resistor') +extract_devices(resistor('res_metal1', 1.0, RES2), { 'R' => res_metal1, 'C' => metal1_con }) + +# RM2 +logger.info('Extracting res_metal2 resistor') +extract_devices(resistor('res_metal2', 1.0, RES2), { 'R' => res_metal2, 'C' => metal2_con }) + +# RM3 +logger.info('Extracting res_metal3 resistor') +extract_devices(resistor('res_metal3', 1.0, RES2), { 'R' => res_metal3, 'C' => metal3_con }) + +# RM4 +logger.info('Extracting res_metal4 resistor') +extract_devices(resistor('res_metal4', 1.0, RES2), { 'R' => res_metal4, 'C' => metal4_con }) + +# RM5 +logger.info('Extracting res_metal5 resistor') +extract_devices(resistor('res_metal5', 1.0, RES2), { 'R' => res_metal5, 'C' => metal5_con }) + +# RTM1 +logger.info('Extracting res_topmetal1 resistor') +extract_devices(resistor('res_topmetal1', 1.0, RES2), { 'R' => res_topmetal1, 'C' => topmetal1_con }) + +# RTM2 +logger.info('Extracting res_topmetal2 resistor') +extract_devices(resistor('res_topmetal2', 1.0, RES2), { 'R' => res_topmetal2, 'C' => topmetal2_con }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_connections.lvs new file mode 100644 index 00000000..2245c9e4 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_connections.lvs @@ -0,0 +1,24 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ---- RF-MOSFET CONNECTIONS ---- +#================================ + +logger.info('Starting LVS RF-MOSFET CONNECTIONS') diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_derivations.lvs new file mode 100644 index 00000000..87a0f0a3 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_derivations.lvs @@ -0,0 +1,47 @@ +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================== +# ----- RF-MOSFET DERIVATIONS ----- +#================================== + +logger.info('Starting RF-MOSFET DERIVATIONS') + +# =============== +# --- RF-NMOS --- +# =============== + +logger.info('Starting RF-NMOS DERIVATIONS') + + +# rfnmos - LV +rfngate_lv = ngate_lv_base.and(rfnmos_mk).not(rfnmos_exc) + +# rfnmos - HV +rfngate_hv = ngate_hv_base.and(rfnmoshv_mk).not(rfnmos_exc) + +# =============== +# --- RF-PMOS --- +# =============== + +logger.info('Starting RF-PMOS DERIVATIONS') + +# rfpmos - LV +rfpgate_lv = pgate_lv_base.and(rfpmos_mk).not(rfpmos_exc) + +# rfpmos - HV +rfpgate_hv = pgate_hv_base.and(rfpmoshv_mk).not(rfpmos_exc) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_extraction.lvs new file mode 100644 index 00000000..37a20a49 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/rfmos_extraction.lvs @@ -0,0 +1,80 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +# ================================ +# ----- RF-MOSFET EXTRACTION ----- +# ================================ + +logger.info('Starting RF-MOSFET EXTRACTION') + +# =============== +# --- RF-NMOS --- +# =============== + +# rfnmos - LV +logger.info('Extraction of RF-NMOS-LV transistor') +extract_devices(mos4('rfnmos'), + { 'SD' => nsd_fet, + 'G' => rfngate_lv, + 'W' => pwell, + 'tS' => nsd_fet, + 'tD' => nsd_fet, + 'tG' => poly_con, + 'tW' => ptap, + }) + +# rfnmos - HV +logger.info('Extraction of RF-NMOS-HV transistor') +extract_devices(mos4('rfnmoshv'), + { 'SD' => nsd_fet, + 'G' => rfngate_hv, + 'W' => pwell, + 'tS' => nsd_fet, + 'tD' => nsd_fet, + 'tG' => poly_con, + 'tW' => ptap, + }) + +# =============== +# --- RF-PMOS --- +# =============== + +# rfpmos - LV +logger.info('Extraction of RF-PMOS-LV transistor') +extract_devices(mos4('rfpmos'), + { 'SD' => psd_fet, + 'G' => rfpgate_lv, + 'tS' => psd_fet, + 'tD' => psd_fet, + 'tG' => poly_con, + 'W' => nwell_drw, + 'tW' => ntap, + }) + +# rfpmos - HV +logger.info('Extraction of RF-PMOS-HV transistor') +extract_devices(mos4('rfpmoshv'), + { 'SD' => psd_fet, + 'G' => rfpgate_hv, + 'tS' => psd_fet, + 'tD' => psd_fet, + 'tG' => poly_con, + 'W' => nwell_drw, + 'tW' => ntap, + }) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_connections.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_connections.lvs new file mode 100644 index 00000000..f922e2a6 --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_connections.lvs @@ -0,0 +1,32 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=============================== +# ------ Taps CONNECTIONS ------ +#=============================== + +logger.info('Starting LVS Taps CONNECTIONS') + +# ntap1 +connect(ntap1_tie, cont_drw) +connect(ntap1_well, nwell_drw) + +# ptap1 +connect(ptap1_tie, cont_drw) +connect(ptap1_sub, pwell) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_derivations.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_derivations.lvs new file mode 100644 index 00000000..43bb65fd --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_derivations.lvs @@ -0,0 +1,42 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#=============================== +# ------ Taps DERIVATIONS ------ +#=============================== + +logger.info('Starting Taps DERIVATIONS') + +taps_exclude = gatpoly.join(nsd_drw).join(trans_drw) + .join(emwind_drw).join(emwihv_drw).join(salblock_drw) + .join(polyres_drw).join(extblock_drw).join(res_drw) + .join(activ_mask).join(recog_diode).join(recog_esd) + .join(ind_drw).join(ind_pin) + +# === ntap1 === +ntap1_exc = pwell.join(psd_drw).join(taps_exclude) + +ntap1_tie = nactiv.and(ntap1_mk).extents.not(ntap1_exc) +ntap1_well = ntap1_mk.covering(ntap1_tie) + +# === ptap1 === +ptap1_exc = nwell_drw.join(taps_exclude) + +ptap1_tie = pactiv.and(ptap1_mk).extents.not(ptap1_exc) +ptap1_sub = pwell.covering(ptap1_tie) diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_extraction.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_extraction.lvs new file mode 100644 index 00000000..d6cc82db --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/rule_decks/tap_extraction.lvs @@ -0,0 +1,32 @@ +# frozen_string_literal: true + +#========================================================================== +# Copyright 2024 IHP PDK Authors +# +# Licensed under the Apache License, Version 2.0 (the "License"); +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# https://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# SPDX-License-Identifier: Apache-2.0 +#========================================================================== + +#================================ +# ------ EFUSE EXTRACTIONS ------ +#================================ + +logger.info('Starting Taps EXTRACTION') + +# ntap1 +logger.info('Extracting ntap1 device') +extract_devices(diode('ntap1', CustomTap), { 'N' => ntap1_tie, 'P' => ntap1_well }) + +# ptap1 +logger.info('Extracting ptap1 device') +extract_devices(diode('ptap1', CustomTap), { 'N' => ptap1_tie, 'P' => ptap1_sub }) diff --git a/pdk/ihp_sg13g2/kpex/sg13g2.lvs b/pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lvs similarity index 100% rename from pdk/ihp_sg13g2/kpex/sg13g2.lvs rename to pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lvs diff --git a/pdk/ihp_sg13g2/kpex/sg13g2.lyp b/pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lyp similarity index 100% rename from pdk/ihp_sg13g2/kpex/sg13g2.lyp rename to pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lyp diff --git a/pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lyt b/pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lyt new file mode 100644 index 00000000..488e57bf --- /dev/null +++ b/pdk/ihp_sg13g2/libs.tech/kpex/sg13g2.lyt @@ -0,0 +1,198 @@ + + + + sg13g2 + IHP SiGe 130nm technology + + 0.001 + + + sg13g2.lyp + true + 0.01,0.005! + + + 1 + true + true + + + true + layer_map() + true + true + + + true + layer_map() + 0.001 + true + #1 + true + #1 + false + #1 + true + OUTLINE + true + PLACEMENT_BLK + true + REGIONS + true + + 0 + true + .PIN + 2 + true + .PIN + 2 + true + .FILL + 5 + true + .OBS + 3 + true + .BLK + 4 + true + .LABEL + 1 + true + .LABEL + 1 + true + + 0 + true + + 0 + VIA_ + true + default + false + sg13g2.map + + + + false + true + true + 64 + 0 + 1 + 0 + DATA + 0 + 0 + BORDER + layer_map() + true + + + 0.001 + 1 + 100 + 100 + 0 + 0 + 0 + false + false + false + true + layer_map() + + + 0 + 0.001 + layer_map() + true + false + + + 1 + 0.001 + layer_map() + true + false + true + + + + + + GDS2 + + true + false + false + false + false + false + 8000 + 32000 + LIB + + + 2 + true + true + 1 + * + false + + + 0 + + + false + false + + + 0 + + true + + + + GatPoly,Cont,Metal1 + Diff,Cont,Metal1 + Metal1,Via1,Metal2 + Metal2,Via2,Metal3 + Metal3,Via3,Metal4 + Metal4,Via4,Metal5 + Metal5,TopVia1,TopMetal1 + TopMetal1,TopVia2,TopMetal2 + SalBlock='28/0' + Activ='1/0-Salblock' + GatPoly='5/0-SalBlock' + Diff='Activ-GatPoly' + Cont='6/0' + Metal1='8/0-8/29' + Via1='19/0' + Metal2='10/0-10/29' + Via2='29/0' + Metal3='30/0-30/29' + Via3='49/0' + Metal4='50/0-50/29' + Via4='66/0' + Metal5='67/0-67/29' + TopVia1='125/0' + TopMetal1='126/0-126/29' + TopVia2='133/0' + TopMetal2='134/0-134/29' + + diff --git a/pdk/sky130A/kpex/debug_sky130.lvs.lylvs b/pdk/sky130A/libs.tech/kpex/debug_sky130.lvs.lylvs similarity index 100% rename from pdk/sky130A/kpex/debug_sky130.lvs.lylvs rename to pdk/sky130A/libs.tech/kpex/debug_sky130.lvs.lylvs diff --git a/pdk/sky130A/kpex/sky130.lvs b/pdk/sky130A/libs.tech/kpex/sky130.lvs similarity index 100% rename from pdk/sky130A/kpex/sky130.lvs rename to pdk/sky130A/libs.tech/kpex/sky130.lvs diff --git a/pdk/sky130A/kpex/sky130A.lyp b/pdk/sky130A/libs.tech/kpex/sky130A.lyp similarity index 100% rename from pdk/sky130A/kpex/sky130A.lyp rename to pdk/sky130A/libs.tech/kpex/sky130A.lyp diff --git a/pdk/sky130A/libs.tech/kpex/sky130A.lyt b/pdk/sky130A/libs.tech/kpex/sky130A.lyt new file mode 100644 index 00000000..031c0c8a --- /dev/null +++ b/pdk/sky130A/libs.tech/kpex/sky130A.lyt @@ -0,0 +1,169 @@ + + + + + sky130 + SkyWater 130nm technology + + 0.001 + + $PDK_ROOT/$PDK/libs.tech/klayout + sky130A.lyp + true + 0.01,0.005! + + + 1 + true + true + + + true + layer_map() + true + true + + + true + layer_map() + 0.001 + true + #1 + true + #1 + false + #1 + true + OUTLINE + true + PLACEMENT_BLK + true + REGIONS + true + .drawing + 0 + true + .pin + 2 + true + .blockage + 3 + true + .blockage + 4 + true + .label + 1 + true + .drawing + 0 + merged.lef + + + false + true + true + 64 + 0 + 1 + 0 + DATA + 0 + 0 + BORDER + layer_map() + true + + + 0.001 + 1 + 100 + 100 + 0 + 0 + 0 + false + false + false + true + layer_map() + + + 0 + 0.001 + layer_map() + true + false + + + 1 + 0.001 + layer_map() + true + false + true + + + + + + + true + false + false + false + false + 8000 + 32000 + LIB + + + 2 + false + false + 1 + * + false + + + 0 + + + false + false + + + 0 + + true + + + + 66/20,66/44,li + li,67/44,met1 + met1,68/44,met2 + met2,69/44,met3 + met3,70/44,met4 + met4,71/44,met5 + met1='67/20+67/5' + met1='68/20+68/5' + met2='69/20+69/5' + met3='70/20+70/5' + met4='71/20+71/5' + met5='72/20+72/5' + + diff --git a/tests/rcx25/rcx25_test.py b/tests/rcx25/rcx25_test.py index a8c471d7..aba703b7 100644 --- a/tests/rcx25/rcx25_test.py +++ b/tests/rcx25/rcx25_test.py @@ -46,6 +46,11 @@ tags = ("PEX", "2.5D", "MAGIC") +def _kpex_pdk_dir() -> str: + return os.path.realpath(os.path.join(__file__, '..', '..', '..', + 'pdk', 'sky130A', 'libs.tech', 'kpex')) + + def _sky130a_testdata_dir() -> str: return os.path.realpath(os.path.join(__file__, '..', '..', '..', 'testdata', 'designs', 'sky130A')) @@ -58,7 +63,7 @@ def _gds(*path_components) -> str: def _save_layout_preview(gds_path: str, output_png_path: str): kdb.Technology.clear_technologies() - default_lyt_path = os.path.abspath(f"{os.environ['PDK_ROOT']}/sky130A/libs.tech/klayout/tech/sky130A.lyt") + default_lyt_path = os.path.abspath(f"{_kpex_pdk_dir()}/sky130A.lyt") tech = kdb.Technology.create_technology('sky130A') tech.load(default_lyt_path)