Source code for gdsfactory.components.tapers.taper_adiabatic

from __future__ import annotations

from collections.abc import Callable
from typing import Any

import numpy as np
import numpy.typing as npt

import gdsfactory as gf
from gdsfactory.path import transition_adiabatic
from gdsfactory.typings import CrossSectionSpec


def neff_TE1550SOI_220nm(w: float) -> float:
    """Returns the effective index of the fundamental TE mode for a 220nm-thick core with 3.45 index, fully clad with 1.44 index.

    Args:
        w: width in um.

    Returns:
        effective index.
    """
    adiabatic_polyfit_TE1550SOI_220nm = np.array(
        [
            1.02478963e-09,
            -8.65556534e-08,
            3.32415694e-06,
            -7.68408985e-05,
            1.19282177e-03,
            -1.31366332e-02,
            1.05721429e-01,
            -6.31057637e-01,
            2.80689677e00,
            -9.26867694e00,
            2.24535191e01,
            -3.90664800e01,
            4.71899278e01,
            -3.74726005e01,
            1.77381560e01,
            -1.12666286e00,
        ]
    )
    return np.poly1d(adiabatic_polyfit_TE1550SOI_220nm)(w)  # type: ignore


[docs] @gf.cell def taper_adiabatic( width1: float = 0.5, width2: float = 5.0, length: float = 0, neff_w: Callable[[float], float] = neff_TE1550SOI_220nm, alpha: float = 1, wavelength: float = 1.55, npoints: int = 200, cross_section: CrossSectionSpec = "strip", max_length: float = 200, ) -> gf.Component: """Returns a straight adiabatic_taper from an effective index callable. Args: width1: initial width. width2: final width. length: 0 uses the optimized length, and otherwise the optimal shape is compressed/stretched to the specified length. neff_w: a callable that returns the effective index as a function of width - By default, will use a compact model of neff(y) for fundamental 1550 nm TE mode of 220nm-thick core with 3.45 index, fully clad with 1.44 index. Many coefficients are needed to capture the behaviour. alpha: parameter that scales the rate of width change. - closer to 0 means longer and more adiabatic; - 1 is the intuitive limit beyond which higher order modes are excited; - [2] reports good performance up to 1.4 for fundamental TE in SOI (for multiple core thicknesses) wavelength: wavelength in um. npoints: number of points for sampling. cross_section: cross_section specification. max_length: maximum length for the taper. References: [1] Burns, W. K., et al. "Optical waveguide parabolic coupling horns." Appl. Phys. Lett., vol. 30, no. 1, 1 Jan. 1977, pp. 28-30, doi:10.1063/1.89199. [2] Fu, Yunfei, et al. "Efficient adiabatic silicon-on-insulator waveguide taper." Photonics Res., vol. 2, no. 3, 1 June 2014, pp. A41-A44, doi:10.1364/PRJ.2.000A41. npoints: number of points for sampling """ xs = gf.get_cross_section(cross_section) layer = xs.layer assert layer is not None # Obtain optimal curve x_opt, w_opt = transition_adiabatic( width1, width2, neff_w=neff_w, wavelength=wavelength, alpha=alpha, max_length=max_length, ) # Resample the points from scipy import interpolate # type: ignore w_opt_interp = interpolate.interp1d(x_opt, w_opt) if not length: length = x_opt[-1] x = np.linspace(0, length, npoints) w: npt.NDArray[np.floating[Any]] = w_opt_interp(x) assert isinstance(w, np.ndarray) # Stretch/compress x x_array: npt.NDArray[np.floating[Any]] = np.linspace(0, length, npoints) * ( 1 + length - x_opt[-1] ) assert isinstance(x_array, np.ndarray) y_array = w / 2 c = gf.Component() c.add_polygon( list(zip(x_array, y_array)) + list(zip(x_array, -y_array))[::-1], # type: ignore[assignment, arg-type] layer=layer, ) # Define ports c.add_port( name="o1", center=(0, 0), width=width1, orientation=180, cross_section=cross_section, layer=layer, ) c.add_port( name="o2", center=(length, 0), width=width2, orientation=0, cross_section=cross_section, layer=layer, ) xs.add_bbox(c) return c
if __name__ == "__main__": c = taper_adiabatic(width1=0.5, width2=5, cross_section="rib_bbox") c.show()