mlt-core 0.9.0

MapLibre Tile library code
Documentation
use std::hint::black_box;

use criterion::{BatchSize, BenchmarkId, Criterion, Throughput, criterion_group, criterion_main};
use mlt_core::__private::{dec, parser};
use mlt_core::Layer;
use mlt_core::encoder::{EncoderConfig, LogicalEncoder, stage_tile};
use strum::IntoEnumIterator as _;

#[path = "bench_utils.rs"]
mod bench_utils;
use bench_utils::{BENCHMARKED_ZOOM_LEVELS, load_mlt_tiles};
use mlt_core::encoder::SortStrategy::Unsorted;
use mlt_core::encoder::{
    Codecs, Encoder, ExplicitEncoder, IntEncoder, PhysicalEncoder, StagedLayer,
};

fn limit<T>(values: impl Iterator<Item = T>) -> impl Iterator<Item = T> {
    if cfg!(debug_assertions) {
        values.take(1)
    } else {
        values.take(usize::MAX)
    }
}

/// Build `StagedLayer` values from decoded tiles for encode benchmarks.
///
/// Goes through `Layer01 → TileLayer → StagedLayer`, which is the correct
/// encode-pipeline entry point per CONTRIBUTING.md.
fn decode_to_owned(tiles: &[(String, Vec<u8>)], tessellate: bool) -> Vec<StagedLayer> {
    tiles
        .iter()
        .flat_map(|(_, data)| {
            let mut d = dec();
            let layers = parser().parse_layers(data).expect("mlt parse failed");
            layers
                .into_iter()
                .filter_map(|layer| {
                    let Layer::Tag01(layer01) = layer else {
                        return None;
                    };
                    let tile = layer01.into_tile(&mut d).ok()?;
                    Some(stage_tile(tile, Unsorted, false, tessellate))
                })
                .collect::<Vec<_>>()
        })
        .collect()
}

fn bench_encode(c: &mut Criterion) {
    let mut group = c.benchmark_group("mlt encode");
    for zoom in BENCHMARKED_ZOOM_LEVELS {
        let tiles = load_mlt_tiles(zoom);
        let total_bytes: usize = tiles.iter().map(|(_, d)| d.len()).sum();
        group.throughput(Throughput::Bytes(total_bytes as u64));
        for tessellate in [true, false] {
            let enc_config = EncoderConfig {
                tessellate,
                ..Default::default()
            };
            for physical in limit(PhysicalEncoder::iter()) {
                for logical in limit(LogicalEncoder::iter()) {
                    let int_enc = IntEncoder::new(logical, physical);
                    group.bench_with_input(
                        BenchmarkId::new(
                            format!("{logical:?}-{physical:?}/tessellate: {tessellate:?}"),
                            zoom,
                        ),
                        &tiles,
                        |b, tiles| {
                            b.iter_batched(
                                || decode_to_owned(tiles, enc_config.tessellate),
                                |layers| {
                                    let mut codecs = Codecs::default();
                                    for layer in layers {
                                        let enc = Encoder::with_explicit(
                                            enc_config,
                                            ExplicitEncoder::all(int_enc),
                                        );
                                        black_box(
                                            layer
                                                .encode_into(enc, &mut codecs)
                                                .expect("encode failed"),
                                        );
                                    }
                                },
                                BatchSize::SmallInput,
                            );
                        },
                    );
                }
            }
        }
    }

    group.finish();
}

criterion_group!(benches, bench_encode);
criterion_main!(benches);