Skip to main content

Oxygen A-band radiative-transfer model with Python bindings

Project description

zdisamar

zdisamar is a Zig implementation of the oxygen A-band radiative-transfer model used in DISAMAR aerosol-layer-height retrieval studies. It calculates top-of-atmosphere reflectance and reflectance derivatives for scenes in which oxygen absorption, aerosol scattering, surface reflection, and instrument spectral response all affect the measured spectrum.

The Fortran DISAMAR code family is the scientific reference for this work. zdisamar keeps the same radiative-transfer problem and reorganizes the repeated oxygen A-band calculations so validation cases can be run, timed, and inspected through generated spectra and timing files.

The Python wrapper is demonstrated in executable notebooks under scripts/demo/. Build the native library first:

zig build

Then open the notebooks:

uv run --with jupyterlab --with ipykernel python -m jupyter lab scripts/demo

The two demos are o2a_plot_bundle.ipynb, which shows the Python-facing O2 A output and plotting accessors, and optimal_estimation_demo.ipynb, which shows a two-state O2 A optimal-estimation flow.

Why The Oxygen A Band

The oxygen A band near 758-770 nm is a strong molecular oxygen absorption band used by passive satellite retrievals to infer information about the vertical placement of scattering layers. Oxygen is well mixed in the atmosphere, so the absorption structure in a measured top-of-atmosphere spectrum carries information about photon path length. Photons scattered by a lower aerosol or cloud layer travel through more oxygen than photons scattered by a higher layer.

This makes the band useful for aerosol-layer-height and cloud-height retrievals. The oxygen absorption signal is also sensitive to surface brightness, geometry, aerosol optical thickness, and the balance between atmospheric and surface contributions to the measured reflectance. Aerosols scatter less strongly than clouds, so aerosol retrievals need a detailed RTM.

Aerosol scene and oxygen A-band reflectance

The figure links a visible aerosol scene to the O2 A reflectance spectrum seen by the instrument: the aerosol contribution changes both the absolute reflectance and the structure inside the absorption band. Aerosol optical thickness, aerosol vertical distribution, and surface reflection all affect the oxygen A-band retrieval.

That spectral change can be used to retrieve atmospheric properties. Light that travels deeper into the atmosphere passes through more oxygen and therefore has deeper absorption-band structure. If photons meet an aerosol or cloud layer higher in the atmosphere, they scatter back toward the instrument earlier and travel through less oxygen. The absorption profile then carries information about where the scattering happened. An RTM makes this usable: for a given atmosphere, surface, viewing geometry, and instrument response, it calculates the reflectance spectrum and the derivatives needed by the retrieval to update the atmospheric state.

What Changed Relative To Fortran DISAMAR

The comparisons use the Fortran DISAMAR code family. Source links in the performance notes point to the KNMI GitLab snapshot d17c52884a875cb87b98e4c4ea7f722659e685ac.

Fortran DISAMAR is the grandfather of this implementation. It is a mature radiative-transfer and retrieval model for passive atmospheric remote sensing: it reads a retrieval configuration, prepares atmospheric and surface inputs, calculates spectra and Jacobians, and runs inverse methods such as optimal estimation. Its strength is breadth. It supports many retrieval families, spectral ranges, configuration options, and operational/research use cases. That breadth also makes focused O2 A benchmarking difficult: a single aerosol-height case still passes through general setup, broad configuration handling, and general numerical routines built for a much wider set of retrieval problems.

Both implementations target the same O2 A retrieval RTM: line-by-line oxygen absorption, multiple scattering, instrument-grid convolution, and reflectance derivatives for optimal estimation. The performance improvements come from reducing repeated setup around that calculation:

  • scene, spectroscopy, geometry, and reference data are loaded once and reused across repeated RTM calls;
  • optimal-estimation retrievals call the RTM several times while the scene, instrument grid, spectroscopy, and many optical inputs stay the same. Each iteration reuses that O2 A calculation state in memory;
  • retrieval Jacobians are calculated for the active state-vector columns;
  • common O2 A LABOS matrix shapes for the 20-stream case use specialized implementations in the repeated layer-doubling calculations;
  • validation and benchmark evidence is stored under validation/outputs/.

The benchmark cases use nstreamsSim = 20 and nstreamsRetr = 20. Streams are the angular quadrature directions used by the multiple-scattering radiative-transfer solver; more streams resolve the angular radiation field more finely, but each RTM call costs more. The production DISAMAR O2 A setup usually uses 16 streams, so these retained 20-stream timings are deliberately slower than a production-tuned Fortran run.

The DISAMAR baseline configuration also keeps aerosolLayerHeight = 0. We do not use the Fortran aerosolLayerHeight = 1 flag to speed the comparison up, because that flag activates an older shortcut path. The timings below therefore compare zdisamar against the normal physical inverse problem, not against a shortcut-accelerated DISAMAR run.

Benchmarks

The benchmark evidence covers RTM timing and optimal-estimation retrieval timing.

RTM

The RTM benchmark calculates one O2 A spectrum over 755-776 nm. The reported spectrum has 701 instrument-grid wavelengths, but each instrument channel is an average over sharper oxygen absorption structure at higher spectral resolution:

low-overhead prepare_o2a       0.057692 s
low-overhead forward elapsed   1.328534 s
ztracy forward elapsed         2.443697 s
output wavelengths                  701
high-resolution radiance samples  3,874
LABOS Fourier terms             120,390
LABOS layer visits            5,417,550
doubling steps                8,389,666

The low-overhead evidence is research/performance/tracing/output/lauka-forward/forward-run/summary.json. The timeline trace summary is research/performance/tracing/output/labos-bottleneck/summary.json. The detailed performance notes live in research/performance/o2a-forward/.

Retrieval

The paired optimal-estimation sweep compares DISAMAR Fortran and zdisamar using the same scene and a-priori sampling. Each system retrieves its own synthetic spectrum, which keeps the retrieval problem aligned while measuring the two systems separately.

DISAMAR Fortran: 100/100 converged, median 1228.826 s, mean 1189.862 s
zdisamar:        100/100 converged, median    3.624 s, mean    3.667 s

Paired optimal-estimation retrieval comparison

The lower row shows the paired retrieved-state difference for each scene, computed as zdisamar retrieved value minus DISAMAR Fortran retrieved value:

aerosol optical depth:       median +1.688e-08, mean -3.025e-07, range -3.703e-05 to +5.423e-06
aerosol mid pressure [hPa]:  median -0.0016,    mean -0.0020,    range -0.0522 to +0.0821

The tracked summary is validation/outputs/optimal_estimation/paired_oe_plot_manifest.json. The retrieval notes live in research/performance/o2a-retrieval/.

Bottlenecks

The oxygen A band contains narrow absorption lines. To model an instrument measurement, zdisamar calculates radiance at high-resolution wavelengths and then applies the instrument spectral response to form the 701 reported wavelengths.

The benchmark expands one spectrum as follows:

701 output wavelengths
-> 3,874 high-resolution radiance samples
-> 120,390 LABOS Fourier terms
-> millions of layer, doubling, and scattering-order operations

The main remaining costs are the repeated LABOS radiative-transfer calculations: Fourier transport, RT-layer construction, layer doubling, scattering-order accumulation, and phase-matrix construction. The detailed timing and operation counts are in research/performance/o2a-forward/remaining-bottlenecks.md.

Production Status And Next Work

The stable implementation today is the O2 A RTM: typed inputs, bundled reference data, the Zig library and CLI helpers, Python wrapper demos, spectra validation, and benchmark artifacts.

The optimal-estimation retrieval code currently supports the aerosol-only two-state retrieval case used by the benchmark evidence. The stable API remains centered on the RTM.

The next work is to reduce the cost of repeated reflectance and derivative calculations during retrievals, while preserving the full O2 A result, and to decide which retrieval functions should become part of the stable API.

Nanda et al. (2019) describes an operational neural-network approach that uses a trained replacement for repeated online radiative-transfer calculations. zdisamar keeps the online full-physics calculation explicit, measurable, and available for validation and further optimization.

Repository Layout

Path Purpose
src/input/ atmosphere, geometry, surface, spectroscopy, instrument, and reference-data inputs
src/forward_model/ RTM internals: optical properties, radiative transfer, instrument-grid calculation, and implementations
src/output/ diagnostic reports and spectrum serialization
src/common/ shared units, errors, interpolation, quadrature, and linear algebra
data/ tracked O2 A bundles and reference assets
tests/ O2 A executable checks
validation/ O2 A compatibility, benchmark, and reference evidence
research/performance/ performance provenance, current benchmark notes, and bottleneck analysis
scripts/demo/ executable Python-facing demo notebooks
docs/ DISAMAR context, O2 A runtime, and reference-data boundary

Build And Verification

Prerequisites:

  • Zig 0.15.2 or newer. The repo declares minimum_zig_version = "0.15.2" in build.zig.zon.
  • uv for Python-based helpers.

Build the library and CLI:

zig build

This produces the CLI helpers at ./zig-out/bin/zdisamar and ./zig-out/bin/zdisamar-o2a-plot-spectrum.

Run the fast local verification loop:

zig build check

Run the broader fast presubmit:

zig build test-fast

Run the full verification baseline:

zig build test

Regenerate the tracked O2 A comparison bundle after changing the O2 A RTM or Jacobian validation outputs:

uv run validation/spectra/validate_spectra.py

For temporary Zig caches, use the ephemeral wrapper:

./scripts/zig-build-ephemeral.sh check
./scripts/zig-build-ephemeral.sh test-fast --summary all

To reclaim space from prior runs:

./scripts/clean-zig-caches.sh

Recommended Reading

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distributions

No source distribution files available for this release.See tutorial on generating distribution archives.

Built Distributions

If you're not sure about the file name format, learn more about wheel file names.

zdisamar-0.0.4-py3-none-win_amd64.whl (1.0 MB view details)

Uploaded Python 3Windows x86-64

zdisamar-0.0.4-py3-none-manylinux1_x86_64.manylinux_2_5_x86_64.whl (752.2 kB view details)

Uploaded Python 3manylinux: glibc 2.5+ x86-64

zdisamar-0.0.4-py3-none-macosx_15_0_x86_64.whl (960.3 kB view details)

Uploaded Python 3macOS 15.0+ x86-64

zdisamar-0.0.4-py3-none-macosx_15_0_arm64.whl (935.7 kB view details)

Uploaded Python 3macOS 15.0+ ARM64

File details

Details for the file zdisamar-0.0.4-py3-none-win_amd64.whl.

File metadata

  • Download URL: zdisamar-0.0.4-py3-none-win_amd64.whl
  • Upload date:
  • Size: 1.0 MB
  • Tags: Python 3, Windows x86-64
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for zdisamar-0.0.4-py3-none-win_amd64.whl
Algorithm Hash digest
SHA256 ded408ef45443c21a03a5100e41cbaf4c58109241a7325d8c67b8e1ca90fc142
MD5 7471ab49e2b155f807ea1267e53eefa8
BLAKE2b-256 05257fdb1e2d98bd138c60aeb779a8aec565091d7d936396f73a10fd47b9f397

See more details on using hashes here.

Provenance

The following attestation bundles were made for zdisamar-0.0.4-py3-none-win_amd64.whl:

Publisher: release.yaml on bout3fiddy/zdisamar

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file zdisamar-0.0.4-py3-none-manylinux1_x86_64.manylinux_2_5_x86_64.whl.

File metadata

File hashes

Hashes for zdisamar-0.0.4-py3-none-manylinux1_x86_64.manylinux_2_5_x86_64.whl
Algorithm Hash digest
SHA256 c0195731b7ac1ba8e9bc992dbc360cf9cfcc9bd15d18c4344061bdb174960235
MD5 2065cfef5798b67ee266f44d54164609
BLAKE2b-256 f30895914e93cc605aa215df221cc58702aea549b926c246664420b0c124c3d8

See more details on using hashes here.

Provenance

The following attestation bundles were made for zdisamar-0.0.4-py3-none-manylinux1_x86_64.manylinux_2_5_x86_64.whl:

Publisher: release.yaml on bout3fiddy/zdisamar

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file zdisamar-0.0.4-py3-none-macosx_15_0_x86_64.whl.

File metadata

File hashes

Hashes for zdisamar-0.0.4-py3-none-macosx_15_0_x86_64.whl
Algorithm Hash digest
SHA256 6b8670bcc5d98b480a4e27cc4151ba3e3f5b64bd1392cc53729810e8fdceae4e
MD5 68e07f99ec9ad387da69796148c7aced
BLAKE2b-256 2408fbdd9b6d60da9b76e497bcc75d4097c2f19d1c1e0391faa49e3f43721980

See more details on using hashes here.

Provenance

The following attestation bundles were made for zdisamar-0.0.4-py3-none-macosx_15_0_x86_64.whl:

Publisher: release.yaml on bout3fiddy/zdisamar

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file zdisamar-0.0.4-py3-none-macosx_15_0_arm64.whl.

File metadata

File hashes

Hashes for zdisamar-0.0.4-py3-none-macosx_15_0_arm64.whl
Algorithm Hash digest
SHA256 03357dfff1f6c1d86caebe9ae7bc461dc8b28a85efa4a590af67b93b1b47f79f
MD5 d90b4b556e62a71cfab70daeee76863c
BLAKE2b-256 7c412b58a070ccf6925db694b3bdc4d3823578f5c225c0ae9849be57ba3d4089

See more details on using hashes here.

Provenance

The following attestation bundles were made for zdisamar-0.0.4-py3-none-macosx_15_0_arm64.whl:

Publisher: release.yaml on bout3fiddy/zdisamar

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page