Source code for gdsfactory.routing.route_ports_to_side

from __future__ import annotations

from typing import Any, Literal, cast

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

import gdsfactory as gf
from gdsfactory import typings
from gdsfactory.component import Component
from gdsfactory.port import flipped
from gdsfactory.routing.route_single import route_single
from gdsfactory.typings import CrossSectionSpec, Ports


def sort_key_west_to_east(port: typings.Port) -> float:
    return port.dx


def sort_key_east_to_west(port: typings.Port) -> float:
    return -port.dx


def sort_key_south_to_north(port: typings.Port) -> float:
    return port.dy


def sort_key_north_to_south(port: typings.Port) -> float:
    return -port.dy


[docs] def route_ports_to_side( component: Component, cross_section: CrossSectionSpec, ports: Ports | None = None, side: Literal["north", "east", "south", "west"] = "north", x: float | None | Literal["east", "west"] = None, y: float | None | Literal["north", "south"] = None, **kwargs: Any, ) -> tuple[list[ManhattanRoute], list[kf.Port]]: """Routes ports to a given side. Args: component: component to route. cross_section: cross_section to use for routing. ports: ports to route to a side. side: 'north', 'south', 'east' or 'west'. x: position to route ports for east/west. None, uses most east/west value. y: position to route ports for south/north. None, uses most north/south value. kwargs: additional arguments to pass to the routing function. Keyword Args: radius: in um. separation: in um. extend_bottom/extend_top for east/west routing. extend_left, extend_right for south/north routing. Returns: List of routes: with routing elements. List of ports: of the new ports. .. plot:: :include-source: import gdsfactory as gf c = gf.Component() dummy = gf.components.nxn(north=2, south=2, west=2, east=2) sides = ["north", "south", "east", "west"] d = 100 positions = [(0, 0), (d, 0), (d, d), (0, d)] for pos, side in zip(positions, sides): dummy_ref = c << dummy dummy_ref.move(pos) routes, ports = gf.routing.route_ports_to_side( component=c, side=side, ports=dummy_ref.ports, cross_section="strip" ) for i, p in enumerate(ports): c.add_port(name=f"{side[0]}{i}", port=p) c.plot() """ if not ports: return [], [] if side in {"north", "south"}: y_value = y if y is not None else side if isinstance(y_value, str): y_value = cast(Literal["north", "south"], y_value) side = cast(Literal["north", "south"], side) return route_ports_to_y( component=component, ports=ports, y=y_value, side=side, cross_section=cross_section, **kwargs, ) elif side in {"east", "west"}: x_value = x if x is not None else side if isinstance(x_value, str): x_value = cast(Literal["east", "west"], x_value) side = cast(Literal["west", "east"], side) return route_ports_to_x( component=component, ports=ports, x=x_value, side=side, cross_section=cross_section, **kwargs, ) else: raise ValueError(f"side={side} must be 'north', 'south', 'east' or 'west'")
[docs] def route_ports_to_x( component: Component, ports: Ports, cross_section: CrossSectionSpec, x: float | Literal["east", "west"] = "east", separation: float = 10.0, radius: float = 10.0, extend_bottom: float = 0.0, extend_top: float = 0.0, extension_length: float = 0.0, y0_bottom: float | None = None, y0_top: float | None = None, backward_port_side_split_index: int = 0, start_straight_length: float = 0.01, dx_start: float | None = None, dy_start: float | None = None, side: Literal["east", "west"] = "east", **routing_func_args: Any, ) -> tuple[list[ManhattanRoute], list[typings.Port]]: """Returns route to x. Args: component: component to route. ports: reasonably well behaved list of ports. ports facing north ports are norther than any other ports ports facing south ports are souther ... ports facing west ports are the wester ... ports facing east ports are the easter ... cross_section: cross_section to use for routing. x: float or string. if float: x coordinate to which the ports will be routed if string: "east" -> route to east if string: "west" -> route to west separation: in um. radius: in um. extend_bottom: in um. extend_top: in um. extension_length: in um. y0_bottom: in um. y0_top: in um. backward_port_side_split_index: integer represents and index in the list of backwards ports (bottom to top) all ports with an index strictly lower or equal are routed bottom all ports with an index larger or equal are routed top. start_straight_length: in um. dx_start: override minimum starting x distance. dy_start: override minimum starting y distance. side: "east" or "west". routing_func_args: additional arguments to pass to the routing function. Returns: routes: list of routes ports: list of the new optical ports 1. routes the bottom-half of the ports facing opposite side of x 2. routes the south ports 3. front ports 4. north ports """ north_ports = [p for p in ports if p.orientation == 90] south_ports = [p for p in ports if p.orientation == 270] east_ports = [p for p in ports if p.orientation == 0] west_ports = [p for p in ports if p.orientation == 180] epsilon = 1.0 a = epsilon + max(radius, separation) bx = epsilon + max(radius, dx_start) if dx_start else a by = epsilon + max(radius, dy_start) if dy_start else a xs = [p.dx for p in ports] ys = [p.dy for p in ports] if y0_bottom is None: y0_bottom = min(ys) - by y0_bottom -= extend_bottom if y0_top is None: y0_top = max(ys) + (max(radius, dy_start) if dy_start else a) y0_top += extend_top if x == "west" and extension_length > 0: extension_length = -extension_length if x == "east": x = max(p.dx for p in ports) + bx elif x == "west": x = min(p.dx for p in ports) - bx if x < min(xs): sort_key_north = sort_key_west_to_east sort_key_south = sort_key_west_to_east forward_ports = west_ports backward_ports = east_ports angle = 0 elif x > max(xs): sort_key_south = sort_key_east_to_west sort_key_north = sort_key_east_to_west forward_ports = east_ports backward_ports = west_ports angle = 180 else: raise ValueError("x should be either to the east or to the west of all ports") # forward_ports.sort() north_ports.sort(key=sort_key_north) south_ports.sort(key=sort_key_south) forward_ports.sort(key=sort_key_south_to_north) backward_ports.sort(key=sort_key_south_to_north) backward_ports_thru_south = backward_ports[:backward_port_side_split_index] backward_ports_thru_north = backward_ports[backward_port_side_split_index:] backward_ports_thru_south.sort(key=sort_key_south_to_north) backward_ports_thru_north.sort(key=sort_key_north_to_south) routes: list[ManhattanRoute] = [] new_ports: list[typings.Port] = [] def add_port( port: typings.Port, y: float, l_elements: list[ManhattanRoute], l_ports: list[typings.Port], start_straight_length: float = start_straight_length, ) -> None: if side == "west": angle = 0 elif side == "east": angle = 180 else: raise ValueError(f"{side=} should be either 'west' or 'east'") new_port = port.copy() new_port.orientation = angle new_port.dx = x + extension_length new_port.dy = y new_port2 = new_port.copy() new_port2.trans *= gf.kdb.Trans.R180 l_elements += [ route_single( component, port, new_port, start_straight_length=start_straight_length, radius=radius, cross_section=cross_section, **routing_func_args, ) ] l_ports.append(new_port2) y_optical_bot = y0_bottom for p in south_ports: add_port(p, y_optical_bot, routes, new_ports) y_optical_bot -= separation for p in forward_ports: add_port(p, p.dy, routes, new_ports) y_optical_top = y0_top for p in north_ports: add_port(p, y_optical_top, routes, new_ports) y_optical_top += separation start_straight_length_section = start_straight_length max_x = max(xs) min_x = min(xs) for p in backward_ports_thru_north: # Extend new_ports if necessary if angle == 0 and p.dx < max_x: start_straight_length_section = max_x - p.dx elif angle == 180 and p.dx > min_x: start_straight_length_section = p.dx - min_x else: start_straight_length_section = 0 add_port( p, y_optical_top, routes, new_ports, start_straight_length=start_straight_length + start_straight_length_section, ) y_optical_top += separation start_straight_length += separation start_straight_length_section = start_straight_length for p in backward_ports_thru_south: # Extend new_ports if necessary if angle == 0 and p.dx < max_x: start_straight_length_section = max_x - p.dx elif angle == 180 and p.dx > min_x: start_straight_length_section = p.dx - min_x else: start_straight_length_section = 0 add_port( p, y_optical_bot, routes, new_ports, start_straight_length=start_straight_length + start_straight_length_section, ) y_optical_bot -= separation start_straight_length += separation return routes, new_ports
[docs] def route_ports_to_y( component: Component, ports: Ports, cross_section: CrossSectionSpec, y: float | Literal["north", "south"] = "north", separation: float = 10.0, radius: float = 10.0, x0_left: float | None = None, x0_right: float | None = None, extension_length: float = 0.0, extend_left: float = 0.0, extend_right: float = 0.0, backward_port_side_split_index: int = 0, start_straight_length: float = 0.01, dx_start: float | None = None, dy_start: float | None = None, side: Literal["north", "south"] = "north", **routing_func_args: Any, ) -> tuple[list[ManhattanRoute], list[typings.Port]]: """Route ports to y. Args: component: component to route. ports: reasonably well behaved list of ports. ports facing north ports are norther than any other ports ports facing south ports are souther ... ports facing west ports are the wester ... ports facing east ports are the easter ... cross_section: cross_section to use for routing. y: float or string. if float: y coordinate to which the ports will be routed if string: "north" -> route to north if string: "south" -> route to south separation: in um. radius: in um. x0_left: in um. x0_right: in um. extension_length: in um. extend_left: in um. extend_right: in um. backward_port_side_split_index: integer this integer represents and index in the list of backwards ports (sorted from left to right) all ports with an index strictly larger are routed right all ports with an index lower or equal are routed left start_straight_length: in um. dx_start: override minimum starting x distance. dy_start: override minimum starting y distance. side: "north" or "south". routing_func_args: additional arguments to pass to the routing function. Returns: - a list of Routes - a list of the new optical ports First route the bottom-half of the back ports (back ports are the one facing opposite side of x) Then route the south ports then the front ports then the north ports """ if y == "south" and extension_length > 0: extension_length = -extension_length da = 45 north_ports = [ p for p in ports if p.orientation > 90 - da and p.orientation < 90 + da ] south_ports = [ p for p in ports if p.orientation > 270 - da and p.orientation < 270 + da ] east_ports = [p for p in ports if p.orientation < da or p.orientation > 360 - da] west_ports = [ p for p in ports if p.orientation < 180 + da and p.orientation > 180 - da ] epsilon = 1.0 a = radius + max(radius, separation) bx = epsilon + max(radius, dx_start) if dx_start else a by = epsilon + max(radius, dy_start) if dy_start else a xs = [p.dx for p in ports] ys = [p.dy for p in ports] if x0_left is None: x0_left = min(xs) - bx x0_left -= extend_left if x0_right is None: x0_right = max(xs) + (max(radius, dx_start) if dx_start else a) x0_right += extend_right if y == "north": y_float = ( max(p.dy + a * np.abs(np.cos(p.orientation * np.pi / 180)) for p in ports) + by ) elif y == "south": y_float = ( min(p.dy - a * np.abs(np.cos(p.orientation * np.pi / 180)) for p in ports) - by ) elif isinstance(y, float | int): y_float = y if y_float <= min(ys): sort_key_east = sort_key_south_to_north sort_key_west = sort_key_south_to_north forward_ports = south_ports backward_ports = north_ports elif y_float >= max(ys): sort_key_west = sort_key_north_to_south sort_key_east = sort_key_north_to_south forward_ports = north_ports backward_ports = south_ports else: raise ValueError("y should be either to the north or to the south of all ports") west_ports.sort(key=sort_key_west) east_ports.sort(key=sort_key_east) forward_ports.sort(key=sort_key_west_to_east) backward_ports.sort(key=sort_key_east_to_west) backward_ports.sort(key=sort_key_west_to_east) backward_ports_thru_west = backward_ports[:backward_port_side_split_index] backward_ports_thru_east = backward_ports[backward_port_side_split_index:] backward_ports_thru_west.sort(key=sort_key_west_to_east) backward_ports_thru_east.sort(key=sort_key_east_to_west) routes: list[ManhattanRoute] = [] new_ports: list[typings.Port] = [] def add_port( port: typings.Port, x: float, l_elements: list[ManhattanRoute], l_ports: list[typings.Port], start_straight_length: float = start_straight_length, ) -> None: if side == "south": angle = 90 elif side == "north": angle = 270 new_port = port.copy() new_port.orientation = angle new_port.dcenter = (x, y_float + extension_length) if np.sum(np.abs((np.array(new_port.center) - port.center) ** 2)) < 1: l_ports += [flipped(new_port)] return try: l_elements += [ route_single( component, port, new_port, start_straight_length=start_straight_length, radius=radius, cross_section=cross_section, **routing_func_args, ) ] l_ports += [flipped(new_port)] except Exception as error: raise ValueError( f"Could not connect {port.name!r} to {new_port.name!r} {error}" ) from error x_optical_left = x0_left for p in west_ports: add_port(p, x_optical_left, routes, new_ports) x_optical_left -= separation for p in forward_ports: add_port(p, p.dx, routes, new_ports) x_optical_right = x0_right for p in east_ports: add_port(p, x_optical_right, routes, new_ports) x_optical_right += separation start_straight_length_section = start_straight_length for p in backward_ports_thru_east: add_port( p, x_optical_right, routes, new_ports, start_straight_length=start_straight_length_section, ) x_optical_right += separation start_straight_length_section += separation start_straight_length_section = start_straight_length for p in backward_ports_thru_west: add_port( p, x_optical_left, routes, new_ports, start_straight_length=start_straight_length_section, ) x_optical_left -= separation start_straight_length_section += separation return routes, new_ports
if __name__ == "__main__": c = Component() cross_section = "strip" dummy = gf.c.nxn(north=2, south=2, west=2, east=2, cross_section=cross_section) dummy_ref = c << dummy # routes, _ = route_ports_to_side( # c, # ports=dummy_ref.ports, # side="south", # cross_section=cross_section, # y=-91, # x=-100, # ) # routes, _ = gf.routing.route_ports_to_side( # c, # ports=dummy_ref.ports, # cross_section=cross_section, # x=50, # side="east", # ) routes, _ = gf.routing.route_ports_to_side( c, ports=dummy_ref.ports, cross_section=cross_section, y=50, side="north", ) # sides = ["north", "south", "east", "west"] # d = 100 # positions = [(0, 0), (d, 0), (d, d), (0, d)] # for pos, side in zip(positions, sides): # dummy_ref = c << dummy # dummy_ref.dcenter = pos # routes = route_ports_to_side(c, dummy_ref.ports, side, layer=(1, 0)) c.show()