797 lines
24 KiB
Python
Executable File
797 lines
24 KiB
Python
Executable File
#!/usr/bin/python3
|
|
|
|
import os
|
|
import sys
|
|
import copy
|
|
|
|
import csv
|
|
|
|
import lxml.etree
|
|
import lxml.builder
|
|
import pprint
|
|
|
|
def isIn(searches, string):
|
|
for search in searches:
|
|
if search in string:
|
|
return True
|
|
return False
|
|
|
|
def startsWithAny(searches, string):
|
|
for search in searches:
|
|
if string.startswith(search):
|
|
return True
|
|
return False
|
|
|
|
def splitLayerID(layerID):
|
|
name = layerID
|
|
purpose = ""
|
|
if "." in name:
|
|
name, purpose = layerID.rsplit(".", 1)
|
|
return name, purpose
|
|
|
|
def purposeToID(purpose):
|
|
if purpose in ["drawing", "dg", "drw"]:
|
|
return "dg"
|
|
elif purpose in ["pin", "pn"]:
|
|
return "pn"
|
|
elif purpose in ["boundary", "by", "bnd"]:
|
|
return "by"
|
|
elif purpose in ["net", "nt"]:
|
|
return "nt"
|
|
elif purpose in ["res", "rs"]:
|
|
return "rs"
|
|
elif purpose in ["label", "ll", "lbl"]:
|
|
return "ll"
|
|
elif purpose in ["cut", "ct"]:
|
|
return "ct"
|
|
elif purpose in ["short", "st", "sho"]:
|
|
return "st"
|
|
elif purpose in ["gate", "ge", "gat"]:
|
|
return "ge"
|
|
elif purpose in ["probe", "pe", "pro"]:
|
|
return "pe"
|
|
elif purpose in ["blockage", "be", "blo"]:
|
|
return "be"
|
|
elif purpose in ["model", "ml", "mod"]:
|
|
return "ml"
|
|
elif startsWithAny(["option", "o", "opt"], purpose):
|
|
return "o"
|
|
elif purpose in ["fuse", "fe", "fus"]:
|
|
return "fe"
|
|
elif purpose in ["mask", "mk"]:
|
|
return "mk"
|
|
elif purpose in ["maskAdd", "md"]:
|
|
return "md"
|
|
elif purpose in ["maskDrop", "mp"]:
|
|
return "mp"
|
|
elif startsWithAny(["waffleAdd", "w"], purpose):
|
|
return "w"
|
|
elif purpose in ["waffleDrop", "wp", "waf"]:
|
|
return "wp"
|
|
elif purpose in ["error", "er", "err"]:
|
|
return "er"
|
|
elif purpose in ["warning", "wg", "wng"]:
|
|
return "wg"
|
|
elif purpose in ["dummy", "dy", "dmy"]:
|
|
return "dy"
|
|
else:
|
|
return "no"
|
|
|
|
def parseLine(line):
|
|
result = list(csv.reader([line.strip()], delimiter=' ', quotechar='"'))[0]
|
|
for i, elem in enumerate(result):
|
|
if elem.startswith("#"):
|
|
result = result[0:i]
|
|
break
|
|
|
|
return [elem.strip() for elem in result if elem.strip()]
|
|
|
|
def loadActConf(path):
|
|
result = dict()
|
|
stack = [result]
|
|
with open(path, "r") as fptr:
|
|
for number, line in enumerate(fptr):
|
|
args = parseLine(line)
|
|
if len(args) > 0:
|
|
if args[0] == "include":
|
|
result = result | loadActConf(args[1])
|
|
elif args[0] == "begin":
|
|
stack[-1][args[1]] = dict()
|
|
stack.append(stack[-1][args[1]])
|
|
elif args[0] == "end":
|
|
stack.pop()
|
|
elif args[0] == "string":
|
|
stack[-1][args[1]] = args[2]
|
|
elif args[0] == "int":
|
|
stack[-1][args[1]] = int(args[2])
|
|
elif args[0] == "real":
|
|
stack[-1][args[1]] = float(args[2])
|
|
elif args[0] == "int_table":
|
|
stack[-1][args[1]] = [int(arg) for arg in args[2:]]
|
|
elif args[0] == "string_table":
|
|
stack[-1][args[1]] = args[2:]
|
|
return result
|
|
|
|
def writeLayerMap(path, conf):
|
|
# See https://github.com/KLayout/klayout/blob/766dd675c11d98b2461c448035197f6e934cb497/src/plugins/streamers/lefdef/db_plugin/dbLEFDEFImporter.cc#L1085
|
|
# Purpose Name = Placement Code
|
|
# LEFPIN = LEFPIN
|
|
# PIN = PIN
|
|
# LEFPINNAME = LEFLABEL
|
|
# PINNAME = LABEL
|
|
# FILL = FILL
|
|
# FILLOPC = FILLOPC
|
|
# LEFOBS = OBS
|
|
# SPNET = SPNET
|
|
# NET = NET
|
|
# VIA = VIA
|
|
# BLOCKAGE = BLK
|
|
# ALL = [LEFPIN, PIN, FILL, FILLOPC, OBS, SPNET, NET, VIA]
|
|
layers = zip(conf["gds"]["layers"], conf["gds"]["major"], conf["gds"]["minor"])
|
|
with open(path, "w") as fptr:
|
|
for layer in layers:
|
|
name, purpose = splitLayerID(layer[0])
|
|
if purpose in ["drawing", "dg", "drw"]:
|
|
print(f"{name} LEFOBS,FILL,FILLOPC,VIA {layer[1]} {layer[2]}", file=fptr)
|
|
elif purpose in ["label", "ll", "lbl"]:
|
|
print(f"NAME {name}/PINNAME {layer[1]} {layer[2]}", file=fptr)
|
|
print(f"NAME {name}/PIN {layer[1]} {layer[2]}", file=fptr)
|
|
print(f"NAME {name}/LEFPINNAME {layer[1]} {layer[2]}", file=fptr)
|
|
print(f"NAME {name}/LEFPIN {layer[1]} {layer[2]}", file=fptr)
|
|
elif purpose in ["net", "nt"]:
|
|
print(f"{name} NET,SPNET {layer[1]} {layer[2]}", file=fptr)
|
|
elif purpose in ["pin", "pin1", "pn"]:
|
|
print(f"{name} PIN,LEFPIN {layer[1]} {layer[2]}", file=fptr)
|
|
elif purpose in ["blockage", "block", "be", "blo"]:
|
|
print(f"{name} BLOCKAGE {layer[1]} {layer[2]}", file=fptr)
|
|
if name.lower().startswith("areaid") and name.lower().endswith("sc"):
|
|
print(f"DIEAREA ALL {layer[1]} {layer[2]}", file=fptr)
|
|
|
|
class Parser(object):
|
|
def __init__(self):
|
|
self.syntax = dict()
|
|
# key -> value
|
|
self.stack = [("", self.syntax)]
|
|
|
|
def start(self, tag, attrs):
|
|
insert = dict()
|
|
if self.stack:
|
|
if tag in self.stack[-1][1]:
|
|
if isinstance(self.stack[-1][1][tag], list):
|
|
self.stack[-1][1][tag].append(insert)
|
|
else:
|
|
self.stack[-1][1][tag] = [
|
|
self.stack[-1][1][tag],
|
|
insert
|
|
]
|
|
else:
|
|
self.stack[-1][1][tag] = insert
|
|
self.stack.append((tag, insert))
|
|
|
|
def end(self, tag):
|
|
if tag == self.stack[-1][0]:
|
|
self.stack.pop()
|
|
|
|
def data(self, data):
|
|
if self.stack and data.strip():
|
|
if len(self.stack) > 1 and isinstance(self.stack[-1][1], dict) and not self.stack[-1][1]:
|
|
if isinstance(self.stack[-2][1][self.stack[-1][0]], list):
|
|
self.stack[-2][1][self.stack[-1][0]][-1] = data
|
|
else:
|
|
self.stack[-2][1][self.stack[-1][0]] = data
|
|
elif isinstance(self.stack[-1][1], str):
|
|
self.stack[-2][1][self.stack[-1][0]] += data
|
|
else:
|
|
print("syntax error", self.stack, data)
|
|
|
|
def close(self):
|
|
return self
|
|
|
|
def readKLayoutConf(path):
|
|
parser = lxml.etree.XMLParser(target = Parser())
|
|
with open(path, "r") as fptr:
|
|
parser.feed(fptr.read())
|
|
return parser.close().syntax
|
|
|
|
def buildKLayoutConf(conf, e=lxml.builder.ElementMaker(), key=None):
|
|
result = []
|
|
if isinstance(conf, dict):
|
|
for key, value in conf.items():
|
|
if not isinstance(value, list):
|
|
value = [value]
|
|
result += buildKLayoutConf(value, e, key)
|
|
elif isinstance(conf, list):
|
|
for item in conf:
|
|
child = e(key)
|
|
elems = buildKLayoutConf(item, e)
|
|
for elem in elems:
|
|
if isinstance(elem, lxml.etree._Element):
|
|
child.append(elem)
|
|
elif elem is not None:
|
|
if isinstance(elem, bool):
|
|
child.text = str(elem).lower()
|
|
else:
|
|
child.text = str(elem)
|
|
result.append(child)
|
|
else:
|
|
result.append(conf)
|
|
return result
|
|
|
|
def writeKLayoutConf(path, conf):
|
|
with open(path, "wb") as fptr:
|
|
klays = buildKLayoutConf(conf)
|
|
for klay in klays:
|
|
fptr.write(lxml.etree.tostring(klay, encoding='utf-8', xml_declaration=True, pretty_print=True))
|
|
|
|
def createLYTFromACT(prs2net, layout, actHome):
|
|
lyt = readKLayoutConf("default.lyt")
|
|
|
|
layers = {
|
|
layer: (major, minor)
|
|
for layer, major, minor in zip(
|
|
layout["gds"]["layers"],
|
|
layout["gds"]["major"],
|
|
layout["gds"]["minor"]
|
|
)
|
|
}
|
|
layerMap = "layer_map(" + ";".join([
|
|
f"'{layer} : {gds[0]}/{gds[1]}'"
|
|
for layer, gds in layers.items()
|
|
]) + ")"
|
|
techName = layout["info"]["name"]
|
|
dbu = float(layout["general"]["scale"])*1e-3
|
|
|
|
# build the tech file (lyt)
|
|
if "technology" not in lyt:
|
|
lyt["technology"] = dict()
|
|
|
|
lyt["technology"] |= {
|
|
"name": techName,
|
|
"description": layout["info"]["date"],
|
|
"dbu": dbu,
|
|
"base-path": f"{actHome}/conf/{techName}/klayout",
|
|
"layer-properties_file": f"{techName}.lyp",
|
|
}
|
|
|
|
if "reader-options" not in lyt["technology"]:
|
|
lyt["technology"]["reader-options"] = dict()
|
|
|
|
if "common" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["common"] = dict()
|
|
|
|
# LEFDEF Layer Purposes
|
|
# from https://github.com/KLayout/klayout/blob/6c8d97adc97bf992ccbdb5f7950cb95a28ffeab9/src/plugins/streamers/lefdef/db_plugin/dbLEFDEFImporter.h#L1037
|
|
# Routing from DEF only
|
|
# Pins from DEF
|
|
# Fills from DEF
|
|
# FillsOPC from DEF
|
|
# SpecialRouting from DEF only
|
|
# LEFPins from LEF
|
|
# ViaGeometry from LEF+DEF
|
|
# Label from DEF
|
|
# LEFLabel from LEF
|
|
# Obstructions from LEF only
|
|
# Outline from LEF+DEF
|
|
# Blockage from DEF only
|
|
# PlacementBlockage from DEF only
|
|
# Regions from DEF only
|
|
# RegionsNone from DEF only
|
|
# RegionsFence from DEF only
|
|
# RegionsGuide from DEF only
|
|
# All from DEF only
|
|
if "lefdef" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["lefdef"] = dict()
|
|
|
|
lyt["technology"]["reader-options"]["lefdef"] |= {
|
|
"layer-map": layerMap,
|
|
"dbu": dbu,
|
|
|
|
# these options are not part of the technology, but are specific to the cell placer
|
|
"produce-placement-blockages": True,
|
|
"placement-blockage-layer": "place.block", # tell the placer to avoid placing cells in an area
|
|
"produce-regions": True,
|
|
"region-layer": "place.mask", # tell the placer to put cells in an area
|
|
|
|
"produce-net-names": True,
|
|
#"net-property-name": "#23", # guessing this is the gds datatype of the *.net layers?
|
|
"produce-inst-names": True,
|
|
#"inst-property-name": "#1", # TODO dump standard cell gds and look for "shape properties"
|
|
"produce-pin-names": True,
|
|
#"pin-property-name": "#1",
|
|
|
|
"produce-cell-outlines": True,
|
|
"cell-outline-layer": "areaid_sc.identifier",
|
|
|
|
"produce-via-geometry": True,
|
|
"via-geometry-suffix-string": ".drawing",
|
|
#"via-geometry-datatype-string": None,
|
|
|
|
"produce-pins": True,
|
|
"pins-suffix-string": ".pin",
|
|
#"pins-datatype-string": None,
|
|
"produce-lef-pins": True,
|
|
"lef_pins-suffix-string": ".pin",
|
|
#"lef_pins-datatype-string": None,
|
|
|
|
"produce-fills": True,
|
|
"fills-suffix-string": ".drawing",
|
|
#"fills-datatype-string": None,
|
|
|
|
"produce-obstructions": True,
|
|
"obstructions-suffix": ".drawing",
|
|
#"obstructions-datatype": None,
|
|
|
|
"produce-blockages": True,
|
|
"blockages-suffix": ".block",
|
|
#"blockages-datatype": None,
|
|
|
|
"produce-labels": True,
|
|
"labels-suffix": ".label",
|
|
#"labels-datatype": None,
|
|
"produce-lef-labels": True,
|
|
"lef-labels-suffix": ".label",
|
|
#"lef-labels-datatype": None,
|
|
|
|
"produce-routing": True,
|
|
"routing-suffix-string": ".drawing",
|
|
#"routing-datatype-string": None,
|
|
"produce-special-routing": True,
|
|
"special-routing-suffix-string": ".drawing",
|
|
#"special-routing-datatype-string": None,
|
|
"via-cellname-prefix": None,
|
|
"lef-files": None,
|
|
}
|
|
|
|
if "mebes" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["mebes"] = dict()
|
|
|
|
if "dxf" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["dxf"] = dict()
|
|
|
|
lyt["technology"]["reader-options"]["dxf"] |= {
|
|
"dbu": dbu,
|
|
"unit": round(prs2net["net"]["lambda"]*1e6/dbu),
|
|
}
|
|
|
|
if "cif" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["cif"] = dict()
|
|
|
|
lyt["technology"]["reader-options"]["cif"] |= {
|
|
"dbu": dbu,
|
|
}
|
|
|
|
if "mag" not in lyt["technology"]["reader-options"]:
|
|
lyt["technology"]["reader-options"]["mag"] = dict()
|
|
|
|
lyt["technology"]["reader-options"]["mag"] |= {
|
|
"dbu": dbu,
|
|
"lambda": round(prs2net["net"]["lambda"]*1e6/dbu),
|
|
}
|
|
|
|
if "connectivity" not in lyt["technology"]:
|
|
lyt["technology"]["connectivity"] = dict()
|
|
|
|
matMap = dict()
|
|
for name, mat in layout["materials"].items():
|
|
if isinstance(mat, dict) and "gds" in mat:
|
|
if name not in matMap:
|
|
matMap[name] = []
|
|
matMap[name] += [layers[layer] for layer in mat["gds"] if layer in layers]
|
|
|
|
metMap = list()
|
|
for name, met in layout["materials"]["metal"].items():
|
|
if name.startswith("m") and name.endswith("_gds"):
|
|
mid = int(name[1:-4])
|
|
if len(metMap) <= mid:
|
|
metMap += [[]]*(mid+1-len(metMap))
|
|
metMap[mid] += [layers[layer] for layer in met if layer in layers]
|
|
|
|
if len(matMap)+len(metMap) > 0:
|
|
if "symbols" not in lyt["technology"]["connectivity"]:
|
|
lyt["technology"]["connectivity"]["symbols"] = list()
|
|
|
|
for name, mat in matMap.items():
|
|
lyt["technology"]["connectivity"]["symbols"].append(
|
|
f"{name}='" + "+".join(f"{gds[0]}/{gds[1]}" for gds in mat) + "'"
|
|
)
|
|
|
|
for mid, met in enumerate(metMap):
|
|
lyt["technology"]["connectivity"]["symbols"].append(
|
|
f"m{mid}='" + "+".join(f"{gds[0]}/{gds[1]}" for gds in met) + "'"
|
|
)
|
|
|
|
if "connection" not in lyt["technology"]["connectivity"]:
|
|
lyt["technology"]["connectivity"]["connection"] = list()
|
|
for name, via in layout["vias"].items():
|
|
if name.endswith("_gds"):
|
|
dn = name[0:-4]
|
|
up = "m1"
|
|
if dn.startswith("m") and dn[1:].isdigit():
|
|
up = int(dn[1:])+1
|
|
|
|
viaMap = [layers[layer] for layer in via if layer in layers]
|
|
lyt["technology"]["connectivity"]["connection"].append(
|
|
f"{dn}," + "+".join(f"{gds[0]}/{gds[1]}" for gds in viaMap) + f",{up}"
|
|
)
|
|
|
|
#pprint.pprint(lyt)
|
|
return lyt
|
|
|
|
def createLYPFromACT(layout, actHome):
|
|
lyp = readKLayoutConf("default.lyp")
|
|
|
|
userPurpose = ["dg", "pn", "ll", "er", "wg"]
|
|
userLayers = ["areaid_sc.identifier", "text.drawing"]
|
|
layers = [(layer, major, minor, purposeToID(splitLayerID(layer)[1]))
|
|
for layer, major, minor in zip(layout["gds"]["layers"], layout["gds"]["major"], layout["gds"]["minor"])]
|
|
layers = sorted([
|
|
layer for layer in layers if layer[0] in userLayers], key=lambda x: (x[1], -x[2])) + sorted([
|
|
layer for layer in layers if layer[3] in userPurpose and layer[0] not in userLayers], key=lambda x: (x[1], -x[2])) + [
|
|
layer for layer in layers if layer[3] not in userPurpose and layer[0] not in userLayers
|
|
]
|
|
|
|
|
|
# build the properties file (lyp)
|
|
if "layer-properties" not in lyp:
|
|
lyp["layer-properties"] = dict()
|
|
|
|
defaultProperties = dict()
|
|
if "properties" in lyp["layer-properties"]:
|
|
defaultProperties = lyp["layer-properties"]["properties"]
|
|
lyp["layer-properties"]["properties"] = []
|
|
|
|
ndiffs = set()
|
|
for diff in layout["diff"]["ntype"]:
|
|
if diff in layout["materials"] and "gds" in layout["materials"][diff]:
|
|
ndiffs.update(layout["materials"][diff]["gds"])
|
|
for well in layout["diff"]["pfet_well"]:
|
|
well = well.split(":")[1]
|
|
if len(well) > 0 and well in layout["materials"] and "gds" in layout["materials"][well]:
|
|
ndiffs.update(layout["materials"][well]["gds"])
|
|
|
|
pdiffs = set()
|
|
for diff in layout["diff"]["ptype"]:
|
|
if diff in layout["materials"] and "gds" in layout["materials"][diff]:
|
|
pdiffs.update(layout["materials"][diff]["gds"])
|
|
for well in layout["diff"]["nfet_well"]:
|
|
well = well.split(":")[1]
|
|
if len(well) > 0 and well in layout["materials"] and "gds" in layout["materials"][well]:
|
|
pdiffs.update(layout["materials"][well]["gds"])
|
|
diffs = ndiffs & pdiffs
|
|
ndiffs = list(ndiffs - diffs)
|
|
pdiffs = list(pdiffs - diffs)
|
|
diffs = list(diffs)
|
|
|
|
diffsSupport = [splitLayerID(layer)[0] for layer in diffs]
|
|
ndiffsSupport = [splitLayerID(layer)[0] for layer in ndiffs]
|
|
pdiffsSupport = [splitLayerID(layer)[0] for layer in pdiffs]
|
|
|
|
pwells = set()
|
|
for well in layout["diff"]["nfet_well"]:
|
|
well = well.split(":")[0]
|
|
if len(well) > 0 and well in layout["materials"] and "gds" in layout["materials"][well]:
|
|
pwells.update(layout["materials"][well]["gds"])
|
|
|
|
nwells = set()
|
|
for well in layout["diff"]["pfet_well"]:
|
|
well = well.split(":")[0]
|
|
if len(well) > 0 and well in layout["materials"] and "gds" in layout["materials"][well]:
|
|
nwells.update(layout["materials"][well]["gds"])
|
|
wells = nwells & pwells
|
|
nwells = list(nwells - wells)
|
|
pwells = list(pwells - wells)
|
|
wells = list(wells)
|
|
|
|
wellsSupport = [splitLayerID(layer)[0] for layer in wells]
|
|
nwellsSupport = [splitLayerID(layer)[0] for layer in nwells]
|
|
pwellsSupport = [splitLayerID(layer)[0] for layer in pwells]
|
|
|
|
poly = set()
|
|
if "polysilicon" in layout["materials"] and "gds" in layout["materials"]["polysilicon"]:
|
|
poly = set(layout["materials"]["polysilicon"]["gds"])
|
|
poly = list(poly)
|
|
polySupport = [splitLayerID(layer)[0] for layer in poly]
|
|
|
|
metals = list()
|
|
if "metals" in layout["general"] and "metal" in layout["materials"]:
|
|
for mid in range(0, layout["general"]["metals"]):
|
|
key = f"m{mid+1}_gds"
|
|
if key in layout["materials"]["metal"]:
|
|
metals.append(layout["materials"]["metal"][key])
|
|
metalsSupport = [[splitLayerID(layer)[0] for layer in metal] for metal in metals]
|
|
metalCol = ["#0000ff", "#ff0080", "#ffa900", "#d700ff", "#00feff", "#13ff00"]
|
|
metalDith = ["I6", "I4", "I8", "I4", "I8", "I4"]
|
|
metalSupportDith = ["I10", "I8", "I4", "I8", "I4", "I8"]
|
|
|
|
viaDiffs = set()
|
|
if "vias" in layout:
|
|
for diff in layout["diff"]["ntype"]:
|
|
if f"{diff}_gds" in layout["vias"]:
|
|
viaDiffs.update(layout["vias"][f"{diff}_gds"])
|
|
for diff in layout["diff"]["ptype"]:
|
|
if f"{diff}_gds" in layout["vias"]:
|
|
viaDiffs.update(layout["vias"][f"{diff}_gds"])
|
|
for well in layout["diff"]["nfet_well"]:
|
|
well = well.split(":")[1]
|
|
if len(well) > 0 and f"{well}_gds" in layout["vias"]:
|
|
viaDiffs.update(layout["vias"][f"{well}_gds"])
|
|
for well in layout["diff"]["pfet_well"]:
|
|
well = well.split(":")[1]
|
|
if len(well) > 0 and f"{well}_gds" in layout["vias"]:
|
|
viaDiffs.update(layout["vias"][f"{well}_gds"])
|
|
viaDiffs = list(viaDiffs)
|
|
viaDiffsSupport = [splitLayerID(layer)[0] for layer in viaDiffs]
|
|
|
|
viaWells = set()
|
|
if "vias" in layout:
|
|
for well in layout["diff"]["nfet_well"]:
|
|
well = well.split(":")[0]
|
|
if len(well) > 0 and f"{well}_gds" in layout["vias"]:
|
|
viaWells.update(layout["vias"][f"{well}_gds"])
|
|
for well in layout["diff"]["pfet_well"]:
|
|
well = well.split(":")[0]
|
|
if len(well) > 0 and f"{well}_gds" in layout["vias"]:
|
|
viaWells.update(layout["vias"][f"{well}_gds"])
|
|
viaWells = list(viaWells)
|
|
viaWellsSupport = [splitLayerID(layer)[0] for layer in viaWells]
|
|
|
|
vias = list()
|
|
if "metals" in layout["general"] and "vias" in layout:
|
|
for mid in range(0, layout["general"]["metals"]-1):
|
|
key = f"m{mid+1}_gds"
|
|
if key in layout["vias"]:
|
|
vias.append(layout["vias"][key])
|
|
viasSupport = [[splitLayerID(layer)[0] for layer in via] for via in vias]
|
|
viaCol = ["#aaaaff", "#ff9acd", "#ffe1a6", "#f2abff", "#b6ffff", "#c9ffc4"]
|
|
|
|
for layer, major, minor, purpose in layers:
|
|
properties = copy.deepcopy(defaultProperties)
|
|
name, purposeFull = splitLayerID(layer)
|
|
|
|
properties |= {
|
|
"name": f"{layer} - {major}/{minor}",
|
|
"source": f"{major}/{minor}",
|
|
"visible": purpose in userPurpose or layer in userLayers,
|
|
}
|
|
|
|
if purpose == "er":
|
|
properties |= {
|
|
"frame-color": "#ff0000",
|
|
"fill-color": "#ff0000",
|
|
"dither-pattern": "blank",
|
|
"line-style": "C",
|
|
"xfill": True,
|
|
}
|
|
elif purpose == "wg":
|
|
properties |= {
|
|
"frame-color": "#ffff00",
|
|
"fill-color": "#ffff00",
|
|
"dither-pattern": "blank",
|
|
"line-style": "C0",
|
|
"xfill": True,
|
|
}
|
|
elif layer in diffs:
|
|
idx = diffs.index(layer)
|
|
properties |= {
|
|
"frame-color": "#ffc280",
|
|
"fill-color": "#ffc280",
|
|
"dither-pattern": "I2",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in ndiffs:
|
|
idx = ndiffs.index(layer)
|
|
properties |= {
|
|
"frame-color": "#80a8ff",
|
|
"fill-color": "#80a8ff",
|
|
"dither-pattern": "I3",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in pdiffs:
|
|
idx = pdiffs.index(layer)
|
|
properties |= {
|
|
"frame-color": "#ff9d9d",
|
|
"fill-color": "#ff9d9d",
|
|
"dither-pattern": "I3",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in wells:
|
|
idx = wells.index(layer)
|
|
properties |= {
|
|
"frame-color": "#ffc280",
|
|
"fill-color": "#ffc280",
|
|
"dither-pattern": "I1",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in nwells:
|
|
idx = nwells.index(layer)
|
|
properties |= {
|
|
"frame-color": "#ff0000",
|
|
"fill-color": "#ff0000",
|
|
"dither-pattern": "I1",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in pwells:
|
|
idx = pwells.index(layer)
|
|
properties |= {
|
|
"frame-color": "#0000ff",
|
|
"fill-color": "#0000ff",
|
|
"dither-pattern": "I1",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in poly:
|
|
idx = poly.index(layer)
|
|
properties |= {
|
|
"frame-color": "#01ff6b",
|
|
"fill-color": "#01ff6b",
|
|
"dither-pattern": "I2",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in set(sum(metals, [])):
|
|
for mid, met in enumerate(metals):
|
|
if layer in met:
|
|
idx = met.index(layer)
|
|
properties |= {
|
|
"frame-color": metalCol[mid%len(metalCol)],
|
|
"fill-color": metalCol[mid%len(metalCol)],
|
|
"dither-pattern": metalDith[mid%len(metalDith)],
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in set(sum(vias, [])):
|
|
for vid, via in enumerate(vias):
|
|
if layer in via:
|
|
idx = via.index(layer)
|
|
properties |= {
|
|
"frame-color": viaCol[vid%len(viaCol)],
|
|
"fill-color": viaCol[vid%len(viaCol)],
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in viaDiffs:
|
|
idx = viaDiffs.index(layer)
|
|
properties |= {
|
|
"frame-color": "#ffffff",
|
|
"fill-color": "#ffffff",
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif layer in viaWells:
|
|
idx = viaWells.index(layer)
|
|
properties |= {
|
|
"frame-color": "#aaffff",
|
|
"fill-color": "#aaffff",
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif purpose == "ll":
|
|
properties |= {
|
|
"frame-color": "#ffffff",
|
|
"fill-color": "#ffffff",
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in diffsSupport:
|
|
idx = diffsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#ffc280",
|
|
"fill-color": "#ffc280",
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in ndiffsSupport:
|
|
idx = ndiffsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#80a8ff",
|
|
"fill-color": "#80a8ff",
|
|
"dither-pattern": "I2",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in pdiffsSupport:
|
|
idx = pdiffsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#ff9d9d",
|
|
"fill-color": "#ff9d9d",
|
|
"dither-pattern": "I2",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in wellsSupport:
|
|
idx = wellsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#ffc280",
|
|
"fill-color": "#ffc280",
|
|
"dither-pattern": "I3",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in nwellsSupport:
|
|
idx = nwellsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#ff0000",
|
|
"fill-color": "#ff0000",
|
|
"dither-pattern": "I3",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in pwellsSupport:
|
|
idx = pwellsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#0000ff",
|
|
"fill-color": "#0000ff",
|
|
"dither-pattern": "I3",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in polySupport:
|
|
idx = polySupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#01ff6b",
|
|
"fill-color": "#01ff6b",
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in viaDiffsSupport:
|
|
idx = viaDiffsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#000000",
|
|
"fill-color": "#ffffff",
|
|
"dither-pattern": "I1",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in viaWellsSupport:
|
|
idx = viaWellsSupport.index(name)
|
|
properties |= {
|
|
"frame-color": "#000000",
|
|
"fill-color": "#aaffff",
|
|
"dither-pattern": "I1",
|
|
"line-style": "C0",
|
|
}
|
|
elif name in set(sum(metalsSupport, [])):
|
|
for mid, met in enumerate(metalsSupport):
|
|
if name in met:
|
|
idx = met.index(name)
|
|
properties |= {
|
|
"frame-color": metalCol[mid%len(metalCol)],
|
|
"fill-color": metalCol[mid%len(metalCol)],
|
|
"dither-pattern": metalSupportDith[mid%len(metalDith)],
|
|
"line-style": "C0",
|
|
}
|
|
elif name in set(sum(viasSupport, [])):
|
|
for vid, via in enumerate(viasSupport):
|
|
if name in via:
|
|
idx = via.index(name)
|
|
properties |= {
|
|
"frame-color": viaCol[vid%len(viaCol)],
|
|
"fill-color": viaCol[vid%len(viaCol)],
|
|
"dither-pattern": "I0",
|
|
"line-style": "C0",
|
|
}
|
|
else:
|
|
pass
|
|
|
|
lyp["layer-properties"]["properties"].append(properties)
|
|
|
|
#pprint.pprint(lyp)
|
|
return lyp
|
|
|
|
def print_help():
|
|
print("Usage: generate_klayout_tech.py [options]")
|
|
print("\t-T<tech>\tidentify the technology used for this translation.")
|
|
|
|
if __name__ == "__main__":
|
|
if len(sys.argv) >= 2 and (sys.argv[1] == '--help' or sys.argv[1] == '-h'):
|
|
print_help()
|
|
else:
|
|
techName = "sky130"
|
|
actHome = os.environ.get('ACT_HOME', "/opt/cad")
|
|
for arg in sys.argv[1:]:
|
|
if arg[0] == '-':
|
|
if arg[1] == 'T':
|
|
techName = arg[2:]
|
|
else:
|
|
print(f"error: unrecognized option '{arg}'")
|
|
print("")
|
|
print_help()
|
|
sys.exit()
|
|
|
|
layout = loadActConf(actHome + "/conf/" + techName + "/layout.conf")
|
|
prs2net = loadActConf(actHome + "/conf/" + techName + "/prs2net.conf")
|
|
writeKLayoutConf(f"{techName}.lyt", createLYTFromACT(prs2net, layout, actHome))
|
|
writeKLayoutConf(f"{techName}.lyp", createLYPFromACT(layout, actHome))
|
|
writeLayerMap("layermap.txt", layout)
|