Source code for kqcircuits.chips.lithography_test

# This code is part of KQCircuits
# Copyright (C) 2021 IQM Finland Oy
#
# This program is free software: you can redistribute it and/or modify it under the terms of the GNU General Public
# License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later
# version.
#
# This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
# warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
#
# You should have received a copy of the GNU General Public License along with this program. If not, see
# https://www.gnu.org/licenses/gpl-3.0.html.
#
# The software distribution should follow IQM trademark policy for open-source software
# (meetiqm.com/iqm-open-source-trademark-policy). IQM welcomes contributions to the code.
# Please see our contribution agreements for individuals (meetiqm.com/iqm-individual-contributor-license-agreement)
# and organizations (meetiqm.com/iqm-organization-contributor-license-agreement).

import numpy

from kqcircuits.pya_resolver import pya

from kqcircuits.chips.chip import Chip
from kqcircuits.test_structures.stripes_test import StripesTest
from kqcircuits.test_structures.stripes_test_increasing_width import StripesTestIncreasingWidth
from kqcircuits.test_structures.cross_test import CrossTest
from kqcircuits.util.parameters import Param, pdt


[docs] class LithographyTest(Chip): """Optical lithography test chip. Consists of StripesTest cells with different parameters. """ stripe_test = Param( pdt.TypeString, "stripe test type", "stripe_test", choices=["stripe_test", "stripe_test_increasing_width"] )
[docs] def build(self): cell_horizontal_1, cell_vertical_1, cell_diagonal_1 = self.create_pattern( num_stripes=20, length=100, min_width=1, max_width=15, step=1, spacing=1, face_id=self.face_ids[0] ) cell_horizontal_2, cell_vertical_2, cell_diagonal_2 = self.create_pattern( num_stripes=20, length=100, min_width=1, max_width=15, step=1, spacing=5, face_id=self.face_ids[0] ) self.insert_cell(cell_horizontal_1, pya.DCplxTrans(1, 0, False, 2000, 6500)) self.insert_cell(cell_horizontal_1, pya.DCplxTrans(1, 0, False, 3000, 6500)) self.insert_cell(cell_horizontal_2, pya.DCplxTrans(1, 0, False, 4000, 6500)) self.insert_cell(cell_horizontal_2, pya.DCplxTrans(1, 0, False, 6000, 6500)) self.insert_cell(cell_vertical_1, pya.DCplxTrans(1, 0, False, 1500, 5700)) self.insert_cell(cell_vertical_1, pya.DCplxTrans(1, 0, False, 5500, 5700)) self.insert_cell(cell_vertical_2, pya.DCplxTrans(1, 0, False, 1500, 3800)) self.insert_cell(cell_vertical_2, pya.DCplxTrans(1, 0, False, 5500, 3800)) self.insert_cell(cell_diagonal_1, pya.DCplxTrans(1, 0, False, 400, 1500)) self.insert_cell(cell_diagonal_1, pya.DCplxTrans(1, 0, False, 1500, 1500)) self.insert_cell(cell_diagonal_2, pya.DCplxTrans(1, 0, False, 3100, 1500)) self.insert_cell(cell_diagonal_2, pya.DCplxTrans(1, 0, False, 6100, 1500))
[docs] def create_pattern(self, num_stripes, length, min_width, max_width, step, spacing, face_id): first_stripes_width = 2 * num_stripes * min_width cell_horizontal = self.layout.create_cell("Stripes") cell_vertical = self.layout.create_cell("Stripes") cell_diagonal = self.layout.create_cell("Stripes") for i, width in enumerate(numpy.arange(min_width, max_width + 0.1 * step, step)): n_stripes = num_stripes if self.stripe_test == "stripe_test": stripes_cell = self.add_element( StripesTest, num_stripes=n_stripes, stripe_width=width, stripe_length=length, stripe_spacing=spacing * width, face_ids=[face_id], ) stripe_row_width = n_stripes * (width + spacing * width) - spacing * width elif self.stripe_test == "stripe_test_increasing_width": n_stripes = int(num_stripes * 0.4) stripes_cell = self.add_element( StripesTestIncreasingWidth, num_stripes=n_stripes, min_stripe_width=width, stripe_step=width, stripe_length=length, stripe_spacing=spacing * width, face_ids=[face_id], ) stripe_row_width = sum(spacing * width + (i + 1) * width for i in range(n_stripes)) # calculate the number of cross alignment marker if (stripe_row_width - 45) % 100 < 50: num_crosses = (stripe_row_width - 45) // 100 + 1 else: num_crosses = (stripe_row_width - 45) // 100 + 2 cross_cell = self.add_element( CrossTest, num_crosses=int(num_crosses), cross_spacing=100, face_ids=[face_id] ) # horizontal cell_horizontal.insert( pya.DCellInstArray( stripes_cell.cell_index(), pya.DCplxTrans(1, 0, False, 0, 2 * i * length + first_stripes_width) ) ) cell_horizontal.insert( pya.DCellInstArray( cross_cell.cell_index(), pya.DCplxTrans( 1, 0, False, 45, 2 * i * length + first_stripes_width - CrossTest.cross_length * 5 / 2 ), ) ) # vertical cell_vertical.insert( pya.DCellInstArray( stripes_cell.cell_index(), pya.DCplxTrans(1, 90, False, 2 * i * length + length + first_stripes_width, 0), ) ) cell_vertical.insert( pya.DCellInstArray( cross_cell.cell_index(), pya.DCplxTrans( 1, 90, True, 2 * i * length + length + first_stripes_width - length - CrossTest.cross_length * 5 / 2, +45, ), ) ) # diagonal diag_offset = 0 # 2*num_stripes*width/numpy.sqrt(8) cell_diagonal.insert( pya.DCellInstArray( stripes_cell.cell_index(), pya.DCplxTrans(1, -45, False, 500 + i * length - diag_offset, 500 + i * length + diag_offset), ) ) return cell_horizontal, cell_vertical, cell_diagonal