Source code for tidy3d.plugins.microwave.impedance_calculator
"""Class for computing characteristic impedance of transmission lines."""
from __future__ import annotations
from typing import Optional, Union
import numpy as np
import pydantic.v1 as pd
from ...components.base import Tidy3dBaseModel
from ...components.data.monitor_data import FieldData, FieldTimeData, ModeSolverData
from ...exceptions import DataError, ValidationError
from .custom_path_integrals import CustomCurrentIntegral2D, CustomVoltageIntegral2D
from .path_integrals import (
CurrentIntegralAxisAligned,
IntegralResultTypes,
MonitorDataTypes,
VoltageIntegralAxisAligned,
)
VoltageIntegralTypes = Union[VoltageIntegralAxisAligned, CustomVoltageIntegral2D]
CurrentIntegralTypes = Union[CurrentIntegralAxisAligned, CustomCurrentIntegral2D]
[docs]
class ImpedanceCalculator(Tidy3dBaseModel):
"""Tool for computing the characteristic impedance of a transmission line."""
voltage_integral: Optional[VoltageIntegralTypes] = pd.Field(
None,
title="Voltage Integral",
description="Definition of path integral for computing voltage.",
)
current_integral: Optional[CurrentIntegralTypes] = pd.Field(
None,
title="Current Integral",
description="Definition of contour integral for computing current.",
)
[docs]
def compute_impedance(self, em_field: MonitorDataTypes) -> IntegralResultTypes:
"""Compute impedance for the supplied ``em_field`` using ``voltage_integral`` and
``current_integral``. If only a single integral has been defined, impedance is
computed using the total flux in ``em_field``.
Parameters
----------
em_field : :class:`.MonitorDataTypes`
The electromagnetic field data that will be used for computing the characteristic
impedance.
Returns
-------
:class:`.IntegralResultTypes`
Result of impedance computation over remaining dimensions (frequency, time, mode indices).
"""
if not isinstance(em_field, (FieldData, FieldTimeData, ModeSolverData)):
raise DataError("'em_field' type not supported by impedance calculator.")
# If both voltage and current integrals have been defined then impedance is computed directly
if self.voltage_integral:
voltage = self.voltage_integral.compute_voltage(em_field)
if self.current_integral:
current = self.current_integral.compute_current(em_field)
# If only one of the integrals has been provided then fall back to using total power (flux)
# with Ohm's law. The input field should cover an area large enough to render the flux
# computation accurate. If the input field is a time signal, then it is real and flux
# corresponds to the instantaneous power. Otherwise the input field is in frequency domain,
# where flux indicates the time-averaged power 0.5*Re(V*conj(I))
if not self.voltage_integral:
flux = em_field.flux
if isinstance(em_field, FieldTimeData):
voltage = flux / current
else:
voltage = 2 * flux / np.conj(current)
if not self.current_integral:
flux = em_field.flux
if isinstance(em_field, FieldTimeData):
current = flux / voltage
else:
current = np.conj(2 * flux / voltage)
impedance = voltage / current
return impedance
[docs]
@pd.validator("current_integral", always=True)
def check_voltage_or_current(cls, val, values):
"""Raise validation error if both ``voltage_integral`` and ``current_integral``
were not provided."""
if not values.get("voltage_integral") and not val:
raise ValidationError(
"Atleast one of 'voltage_integral' or 'current_integral' must be provided."
)
return val