Source code for kqcircuits.elements.waveguide_coplanar_curved

# 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/developers/osstmpolicy). IQM welcomes contributions to the code. Please see our contribution agreements
# for individuals (meetiqm.com/developers/clas/individual) and organizations (meetiqm.com/developers/clas/organization).


from math import pi, sin, cos

from kqcircuits.elements.element import Element
from kqcircuits.pya_resolver import pya
from kqcircuits.util.geometry_helper import vector_length_and_direction
from kqcircuits.util.parameters import Param, pdt, add_parameters_from
from kqcircuits.elements.waveguide_coplanar_straight import WaveguideCoplanarStraight


[docs] def arc(r, start, stop, n): """Returns list of points of an arc Args: r: radius start: begin angle in radians stop: end angle in radians n: number of corners in full circle .. MARKERS_FOR_PNG 0,91 0,100 """ n_steps = max(round(abs(stop - start) * n / (2 * pi)), 1) step = (stop - start) / n_steps r_corner = r / cos(step / 2) pts = [pya.DPoint(r * cos(start), r * sin(start))] for i in range(n_steps): alpha = start + step * (i + 0.5) pts.append(pya.DPoint(r_corner * cos(alpha), r_corner * sin(alpha))) pts.append(pya.DPoint(r * cos(stop), r * sin(stop))) return pts
[docs] @add_parameters_from(WaveguideCoplanarStraight, "add_metal", "ground_grid_in_trace") class WaveguideCoplanarCurved(Element): """The PCell declaration of a curved segment of a coplanar waveguide. Coordinate origin is left at the center of the arc. """ alpha = Param(pdt.TypeDouble, "Curve angle (rad)", pi) length = Param(pdt.TypeDouble, "Actual length", 0, unit="μm", readonly=True)
[docs] def coerce_parameters_impl(self): # Update length self.length = self.r * abs(self.alpha)
[docs] def build(self): ( left_inner_arc, left_outer_arc, right_inner_arc, right_outer_arc, left_protection_arc, right_protection_arc, annotation_arc, ) = WaveguideCoplanarCurved.create_curve_arcs(self, self.alpha) # Left gap pts = left_inner_arc + left_outer_arc shape_1 = pya.DPolygon(pts) self.cell.shapes(self.get_layer("base_metal_gap_wo_grid")).insert(shape_1) # Right gap pts = right_inner_arc + right_outer_arc shape_2 = pya.DPolygon(pts) self.cell.shapes(self.get_layer("base_metal_gap_wo_grid")).insert(shape_2) pts = annotation_arc shape = pya.DPath(pts, self.a) self.cell.shapes(self.get_layer("waveguide_path")).insert(shape) if self.add_metal: self.cell.shapes(self.get_layer("base_metal_addition")).insert(shape) # Protection layer if self.ground_grid_in_trace: self.add_protection(shape_1.sized(1)) self.add_protection(shape_2.sized(1)) else: # Protection layer pts = left_protection_arc + right_protection_arc self.add_protection(pya.DPolygon(pts))
[docs] @staticmethod def create_curve_arcs(elem, angle): """Creates arcs of points for a curved waveguide. Args: elem: Element from which the waveguide parameters for the arc are taken angle (double): angle of the curved waveguide Returns: A tuple consisting of lists of points, each list representing one of the arcs. (left_gap_inner, left_gap_outer, right_gap_inner, right_gap_outer, left_protection, right_protection, annotation) """ alphastart = 0 alphastop = angle left_gap_inner = arc(elem.r - elem.a / 2, alphastart, alphastop, elem.n) left_gap_outer = arc(elem.r - elem.a / 2 - elem.b, alphastop, alphastart, elem.n) right_gap_inner = arc(elem.r + elem.a / 2, alphastart, alphastop, elem.n) right_gap_outer = arc(elem.r + elem.a / 2 + elem.b, alphastop, alphastart, elem.n) left_protection = arc(elem.r - elem.a / 2 - elem.b - elem.margin, alphastart, alphastop, elem.n) right_protection = arc(elem.r + elem.a / 2 + elem.b + elem.margin, alphastop, alphastart, elem.n) annotation = arc(elem.r, alphastart, alphastop, elem.n) return ( left_gap_inner, left_gap_outer, right_gap_inner, right_gap_outer, left_protection, right_protection, annotation, )
[docs] @staticmethod def produce_curve_termination(elem, angle, term_len, trans, face_index=0): """Produces termination for a curved waveguide. The termination consists of a rectangular polygon in the metal gap layer, and grid avoidance around it. The termination is placed at the position where a curved waveguide with alpha=angle and trans=trans would end. Args: elem: Element from which the waveguide parameters for the termination are taken angle (double): angle of the curved waveguide term_len (double): termination length, assumed positive trans (DTrans): transformation applied to the termination face_index (int): face index of the face in elem where the termination is created """ ( left_inner_arc, left_outer_arc, right_inner_arc, right_outer_arc, left_protection_arc, right_protection_arc, _, ) = WaveguideCoplanarCurved.create_curve_arcs(elem, angle) # direction of the termination box _, term_dir = vector_length_and_direction(left_outer_arc[0] - left_outer_arc[1]) if term_len > 0: # metal gap for termination pts = [ left_inner_arc[-1], left_outer_arc[0], left_outer_arc[0] + term_len * term_dir, right_outer_arc[0] + term_len * term_dir, right_outer_arc[0], right_inner_arc[-1], ] shape = pya.DPolygon(pts) elem.cell.shapes(elem.layout.layer(elem.face(face_index)["base_metal_gap_wo_grid"])).insert(trans * shape) # grid avoidance for termination protection_pts = [ left_protection_arc[-1], left_protection_arc[-1] + (term_len + elem.margin) * term_dir, right_protection_arc[0] + (term_len + elem.margin) * term_dir, right_protection_arc[0], ] elem.add_protection(trans * pya.DPolygon(protection_pts), face_index)