Model Comparison to Qucs-S#

This notebook compares the S-parameter models from qpdk (using sax) against reference results from Qucs-S simulations. Each test suite validates a different component model.

The comparisons include:

  • Polar plots showing S-parameters in the complex plane

  • Magnitude and phase plots versus frequency

  • Visual validation of model accuracy against reference data

Hide code cell source

import sys

if "google.colab" in sys.modules:
    import subprocess

    print("Running in Google Colab. Installing QPDK...")
    subprocess.check_call([
        sys.executable,
        "-m",
        "pip",
        "install",
        "-q",
        "qpdk[models] @ git+https://github.com/gdsfactory/quantum-rf-pdk.git",
    ])

Hide code cell source

import importlib.util
import inspect

import numpy as np
from IPython.display import Markdown, display

from qpdk import PDK
from qpdk.config import PATH as QPDK_PATH

# Load the test module directly by file path (avoids sys.path manipulation)
_spec = importlib.util.spec_from_file_location(
    "test_compare_to_qucs",
    QPDK_PATH.tests / "models" / "test_compare_to_qucs.py",
)
_test_module = importlib.util.module_from_spec(_spec)
_spec.loader.exec_module(_test_module)
BaseCompareToQucs = _test_module.BaseCompareToQucs

PDK.activate()

Discover Test Suites#

Dynamically discover all test suite classes that compare qpdk models to Qucs-S results. We find all classes that:

  1. Are subclasses of BaseCompareToQucs

  2. Are not the base class itself

  3. Are concrete classes (not abstract)

Hide code cell source

def discover_test_suites() -> list[type[BaseCompareToQucs]]:
    """Discover all test suite classes for Qucs-S comparison.

    Returns:
        List of test suite classes that inherit from :class:`~BaseCompareToQucs`.
    """
    test_suites = []

    # Get all members of the module (loaded via importlib above)
    for _name, obj in inspect.getmembers(_test_module):
        # Check if it's a class
        if not inspect.isclass(obj):
            continue

        # Check if it's a subclass of BaseCompareToQucs but not the base class itself
        if not issubclass(obj, BaseCompareToQucs) or obj is BaseCompareToQucs:
            continue

        # Check if it's a concrete class (not abstract)
        if inspect.isabstract(obj):
            continue

        test_suites.append(obj)

    return test_suites
# Discover all available test suites
test_suites = discover_test_suites()
print(f"Found {len(test_suites)} test suite(s):")
for suite in test_suites:
    print(f"\t· {suite.__name__}")
Found 5 test suite(s):
	· TestCPWCompareToQucs
	· TestCapacitorCompareToQucs
	· TestCouplerStraightCompareToQucs
	· TestInductorCompareToQucs
	· TestLCResonatorCompareToQucs

Model Comparison#

Compare the S-parameter models against Qucs-S reference data.

# Find and plot all test suites

for suite in test_suites:
    test_instance = suite()
    display(Markdown(f"### {test_instance.component_name}"))
    display(Markdown(f"**Test Suite:** `{suite.__name__}`"))

    param_strs = [
        f"{p.name} = {p.value / p.unit:.2f} × 10^{int(np.log10(p.unit))}"
        for p in sorted(test_instance.parameters, key=lambda x: x.name)
    ]
    param_text = "**Parameters:**\n" + "\n".join(f"- {p}" for p in param_strs)
    display(Markdown(param_text))

    display(Markdown(f"**CSV:** `{test_instance.csv_filename}`"))
    test_instance.plot_comparison()

Coplanar waveguide

Test Suite: TestCPWCompareToQucs

Parameters:

  • length = 10000000000.00 × 10^-6

CSV: cpw_w10_s_6_l10mm.csv

../_images/36631093b95e4e37b7537d3646370861d7ef42d9f6aa2ef8e8d51a8df8a10091.svg

Capacitor

Test Suite: TestCapacitorCompareToQucs

Parameters:

  • capacitance = 60.00 × 10^-15

CSV: capacitor_qucs.csv

../_images/3d476e889f14cb0f24ccbeef3bb85ff6d25bed8d98a24ec9a0569b7490589e7b.svg

Coupler Straight

Test Suite: TestCouplerStraightCompareToQucs

Parameters:

  • gap = 1520000.00 × 10^-6

  • length = 500000000.00 × 10^-6

CSV: coupler_straight_qucs.csv

../_images/5c012ecd8f109ff60199f0b69d8622cac1b0ce442de64d607e56b4178902d99e.svg

Inductor

Test Suite: TestInductorCompareToQucs

Parameters:

  • inductance = 10.00 × 10^-9

CSV: inductor_qucs.csv

../_images/27c0d08943e83b45404ea38503fe76a0423b1a5820f450414a76c9ab3d825ae4.svg

LC Resonator

Test Suite: TestLCResonatorCompareToQucs

Parameters:

  • capacitance = 10.00 × 10^-15

  • inductance = 10.00 × 10^-9

CSV: lc_resonator_qucs.csv

../_images/068bb683be23dd6c4052ebccd6e6ad706672e190b7d848cbd1b91eed0a0f02db.svg

Summary#

The plots above show comparisons between qpdk models (dashed lines) and Qucs-S reference simulations (solid lines) for various passive components:

  • Left plot: Polar representation showing S-parameters in the complex plane

  • Right plot: Magnitude (in dB) and phase (in radians) versus frequency

Good agreement between the models validates the accuracy of the qpdk implementations for use in circuit simulations and design optimization.