jxl-render 0.12.4

JPEG XL image renderer, part of jxl-oxide
Documentation
use jxl_grid::{AlignedGrid, AllocTracker, SharedSubgrid};
use jxl_modular::ChannelShift;

use crate::{Region, Result};

pub fn apply_jpeg_upsampling_single(
    grid: SharedSubgrid<f32>,
    shift: ChannelShift,
    target_region: Region,
    tracker: Option<&AllocTracker>,
) -> Result<AlignedGrid<f32>> {
    fn interpolate(left: f32, center: f32, right: f32) -> (f32, f32) {
        (0.25 * left + 0.75 * center, 0.75 * center + 0.25 * right)
    }

    let height = grid.height();
    let target_width = target_region.width as usize;
    let target_height = target_region.height as usize;

    let mut out = AlignedGrid::with_alloc_tracker(target_width, target_height, tracker)?;
    let buf = out.buf_mut();

    let h_upsampled = shift.hshift() == 0;
    let v_upsampled = shift.vshift() == 0;

    for (y, out_row) in buf.chunks_exact_mut(target_width).take(height).enumerate() {
        let row = grid.get_row(y);
        if h_upsampled {
            out_row.copy_from_slice(&row[..target_width]);
            continue;
        }

        let last_sample = *row.last().unwrap();
        let last_item = [last_sample; 2];
        let adj_samples = row.windows(2).chain(std::iter::once(last_item.as_ref()));

        let mut prev_sample = *row.first().unwrap();
        let it = adj_samples.zip(out_row.chunks_mut(2));
        for (input, output) in it {
            let [curr, next] = *input else { unreachable!() };
            let (left, right) = interpolate(prev_sample, curr, next);
            match output {
                [a] => {
                    *a = left;
                }
                [a, b] => {
                    *a = left;
                    *b = right;
                }
                _ => unreachable!(),
            }
            prev_sample = curr;
        }
    }

    // image is horizontally upsampled here
    if !v_upsampled {
        let mut prev_row = buf[(height - 1) * target_width..][..target_width].to_vec();
        for y in (0..height).rev() {
            let idx_base = y * target_width;
            let top_base = idx_base.saturating_sub(target_width);
            for x in 0..target_width {
                let curr_sample = buf[idx_base + x];

                // We're interpolating bottom-to-top.
                let (bottom, top) = interpolate(prev_row[x], curr_sample, buf[top_base + x]);
                buf[idx_base * 2 + x] = top;
                if y * 2 + 1 < target_height {
                    buf[idx_base * 2 + target_width + x] = bottom;
                }

                prev_row[x] = curr_sample;
            }
        }
    }

    Ok(out)
}