Source code for gdsfactory.routing.route_south

from __future__ import annotations

from collections.abc import Sequence
from functools import partial

import numpy as np
from kfactory.routing.generic import ManhattanRoute

import gdsfactory as gf
from gdsfactory import typings
from gdsfactory.component import Component, ComponentReference
from gdsfactory.cross_section import CrossSection
from gdsfactory.port import Port, select_ports_optical
from gdsfactory.routing.auto_taper import add_auto_tapers
from gdsfactory.routing.route_single import route_single
from gdsfactory.routing.utils import direction_ports_from_list_ports
from gdsfactory.typings import (
    ComponentSpec,
    CrossSectionSpec,
    Radius,
    SelectPorts,
    Strs,
)


[docs] def route_south( component: Component, component_to_route: Component | ComponentReference, optical_routing_type: int = 1, excluded_ports: Sequence[str] | None = None, straight_separation: float = 4.0, io_gratings_lines: list[list[ComponentReference]] | None = None, gc_port_name: str = "o1", bend: ComponentSpec = "bend_euler", straight: ComponentSpec = "straight", select_ports: SelectPorts = select_ports_optical, port_names: Strs | None = None, cross_section: CrossSectionSpec = "strip", start_straight_length: float = 0.5, port_type: str | None = None, allow_width_mismatch: bool = False, auto_taper: bool = True, ) -> list[ManhattanRoute]: """Places routes to route a component ports to the south. Args: component: top level component to add the routes. component_to_route: component or reference to route ports to south. optical_routing_type: routing heuristic `1` or `2` \ 1: uses the component size info to estimate the box size.\ 2: only looks at the optical port positions to estimate the size. excluded_ports: list of port names to NOT route. straight_separation: in um. io_gratings_lines: list of ports to which the ports produced by this function will be connected. \ Supplying this information helps avoiding straight collisions. gc_port_name: grating coupler port name. Used only if io_gratings_lines is supplied. bend: spec. straight: spec. select_ports: function to select_ports. port_names: optional port names. Overrides select_ports. cross_section: cross_section spec. start_straight_length: in um. port_type: optical or electrical. allow_width_mismatch: allow width mismatch. auto_taper: auto taper. Works well if the component looks roughly like a rectangular box with: north ports on the north of the box. south ports on the south of the box. east ports on the east of the box. west ports on the west of the box. .. plot:: :include-source: import gdsfactory as gf c = gf.Component() ref = c << gf.components.ring_double() r = gf.routing.route_south(c, ref) c.plot() """ xs = gf.get_cross_section(cross_section) excluded_ports = excluded_ports or () start_straight_length0 = start_straight_length routes: list[ManhattanRoute] = [] if optical_routing_type not in {1, 2}: raise ValueError( f"optical_routing_type = {optical_routing_type} not in supported [1, 2]" ) if port_names: optical_ports = [component_to_route[port_name] for port_name in port_names] else: optical_ports = [ p for p in select_ports(component_to_route.ports) if p.name not in excluded_ports ] if auto_taper: optical_ports = add_auto_tapers(component, optical_ports, cross_section) if not optical_ports: return [] port_type = port_type or optical_ports[0].port_type bend90 = bend(cross_section=cross_section) if callable(bend) else bend bend90 = gf.get_component(bend90) dy = abs(bend90.info["dy"]) # Handle empty list gracefully route_single_with_conn_params = partial( route_single, bend=bend, straight=straight, cross_section=cross_section, port_type=port_type, allow_width_mismatch=allow_width_mismatch, auto_taper=False, ) # Used to avoid crossing between straights in special cases # This could happen when abs(x_port - x_grating) <= 2 * dy delta_gr_min = 2 * dy + 1 sep = straight_separation # Get lists of optical ports by orientation direction_ports = direction_ports_from_list_ports(optical_ports) north_ports = direction_ports["N"] north_start = north_ports[: len(north_ports) // 2] north_finish = north_ports[len(north_ports) // 2 :] west_ports = direction_ports["W"] west_ports.reverse() east_ports = direction_ports["E"] south_ports = direction_ports["S"] north_finish.reverse() # Sort right to left north_start.reverse() # Sort right to left ordered_ports = north_start + west_ports + south_ports + east_ports + north_finish def get_index_port_closest_to_x( x: float, component_references: list[ComponentReference] ) -> np.intp: return np.array( [abs(x - p.ports[gc_port_name].x) for p in component_references] ).argmin() def gen_port_from_port( x: float, y: float, p: typings.Port, cross_section: CrossSection ) -> Port: return Port( name=p.name, center=(x, y), orientation=90.0, width=p.width, layer=gf.get_layer(cross_section.layer), port_type=p.port_type, ) west_ports.reverse() y0 = min(p.y for p in ordered_ports) - dy - 0.5 ports_to_route: list[Port] = [] optical_xs_tmp = [p.x for p in ordered_ports] x_optical_min = min(optical_xs_tmp) x_optical_max = max(optical_xs_tmp) # Set starting ``x`` on the west side # ``x`` is the x-coord of the waypoint where the current component port is connected. # x starts as close as possible to the component. # For each new port, the distance is increased by the separation. # The starting x depends on the heuristic chosen : ``1`` or ``2`` if optical_routing_type == 1: # use component size to know how far to route x = component_to_route.xmin - dy - 1 else: # use optical port to know how far to route x = x_optical_min - dy - 1 # First route the ports facing west # In case we have to connect these ports to a line of gratings, # Ensure that the port is aligned with the grating port or # has enough space for manhattan routing (at least two bend radius) for p in west_ports: if io_gratings_lines: i_grating = get_index_port_closest_to_x(x, io_gratings_lines[-1]) x_gr = io_gratings_lines[-1][i_grating].ports[gc_port_name].x if abs(x - x_gr) < delta_gr_min: if x > x_gr: x = x_gr elif x < x_gr: x = x_gr - delta_gr_min tmp_port = gen_port_from_port(x, y0, p, cross_section=xs) ports_to_route.append(tmp_port) route = route_single_with_conn_params(component, tmp_port, p) x -= sep # route first halft of north ports above the top west one north_start.reverse() # We need them from left to right start_straight_length = start_straight_length0 if len(north_start) > 0: y_max = max(p.y for p in west_ports + north_start) for p in north_start: tmp_port = gen_port_from_port(x, y0, p, cross_section=xs) route = route_single_with_conn_params( component, port2=p, port1=tmp_port, end_straight_length=start_straight_length + y_max - p.y, ) ports_to_route.append(tmp_port) x -= sep start_straight_length += sep routes.append(route) # Set starting ``x`` on the east side if optical_routing_type == 1: # use component size to know how far to route x = component_to_route.xmax + dy + 1 elif optical_routing_type == 2: # use optical port to know how far to route x = x_optical_max + dy + 1 # Route the east ports # In case we have to connect these ports to a line of gratings, # Ensure that the port is aligned with the grating port or # has enough space for manhattan routing (at least two bend radius) start_straight_length = start_straight_length0 for p in east_ports: if io_gratings_lines: i_grating = get_index_port_closest_to_x(x, io_gratings_lines[-1]) x_gr = io_gratings_lines[-1][i_grating].ports[gc_port_name].x if abs(x - x_gr) < delta_gr_min: if x < x_gr: x = x_gr elif x > x_gr: x = x_gr + delta_gr_min tmp_port = gen_port_from_port(x, y0, p, cross_section=xs) route = route_single_with_conn_params( component, tmp_port, p, end_straight_length=start_straight_length, ) routes.append(route) ports_to_route.append(tmp_port) x += sep # Route the remaining north ports start_straight_length = start_straight_length0 if len(north_finish) > 0: y_max = max(p.y for p in east_ports + north_finish) for p in north_finish: tmp_port = gen_port_from_port(x, y0, p, cross_section=xs) ports_to_route.append(tmp_port) route = route_single_with_conn_params( component, tmp_port, p, end_straight_length=start_straight_length + y_max - p.y, ) x += sep start_straight_length += sep routes.append(route) flipped_ports = [p.copy() for p in ports_to_route] for p in flipped_ports: p.trans *= gf.kdb.Trans.R180 component.add_ports(flipped_ports) component.add_ports(south_ports) component.auto_rename_ports() return routes
if __name__ == "__main__": c = gf.Component() @gf.cell def mzi_with_bend(radius: Radius = 10) -> Component: c = gf.Component() bend = c.add_ref(gf.components.bend_euler(radius=radius)) mzi = c.add_ref(gf.components.mzi()) bend.connect("o1", mzi.ports["o2"]) c.add_port(name="o1", port=mzi.ports["o1"]) c.add_port(name="o2", port=bend.ports["o2"]) return c component = gf.c.mzi_phase_shifter() component = mzi_with_bend() component = gf.components.mmi2x2() component = gf.components.nxn(north=4, south=2, west=2, east=2) component = gf.components.straight(length=10, width=2) ref = c << component # r = route_south(c, ref, optical_routing_type=1, start_straight_length=0) r = route_south(c, ref, auto_taper=True) # print(r.lengths) c.show()