blob: efabfb51a598db6818259c5a0a8d3be331483ba2 [file] [edit]
// Copyright 2026 the Vello Authors
// SPDX-License-Identifier: Apache-2.0 OR MIT
//! Cache key for glyph bitmaps stored in the atlas.
//!
//! [`GlyphCacheKey`] captures every parameter that affects the visual appearance
//! of a rasterized glyph — font identity, size, hinting, subpixel position,
//! COLR context color, and variable-font coordinates. Two keys that compare
//! equal produce identical bitmaps and can safely share a single atlas entry.
use crate::color::{AlphaColor, Srgb};
use crate::glyph::FontEmbolden;
use crate::kurbo::Join;
use core::hash::{Hash, Hasher};
#[cfg(not(feature = "std"))]
use core_maths::CoreFloat as _;
use skrifa::instance::NormalizedCoord;
use smallvec::SmallVec;
/// Number of horizontal subpixel quantization buckets (valid range: 1–253).
///
/// Higher values improve rendering quality at the cost of more atlas entries
/// per glyph. Common values: 1 (disabled), 2, 4 (default), 8.
pub(crate) const SUBPIXEL_BUCKETS: u8 = 4;
/// Sentinel `subpixel_x` for COLR glyph cache entries.
///
/// `quantize_subpixel` returns values in `0..SUBPIXEL_BUCKETS`, so values
/// above that range can never appear in an outline key. Using distinct
/// sentinels for COLR and bitmap entries prevents cache collisions between
/// glyph types that would otherwise produce identical keys (same font, glyph
/// id, size, and color).
pub(crate) const SUBPIXEL_COLR: u8 = SUBPIXEL_BUCKETS;
/// Sentinel `subpixel_x` for bitmap glyph cache entries. See [`SUBPIXEL_COLR`].
pub(crate) const SUBPIXEL_BITMAP: u8 = SUBPIXEL_BUCKETS + 1;
/// Unique identifier for a cached glyph bitmap.
///
/// Two glyphs with the same key are visually identical and can share
/// the same cached bitmap. The key includes all parameters that affect
/// the glyph's appearance.
///
/// `var_coords` is deliberately excluded from `Hash`/`Eq` because the
/// [`GlyphAtlas`](crate::atlas::cache::GlyphAtlas) uses a two-level map
/// structure that already partitions entries by variation coordinates.
/// Callers that use a flat map must ensure equivalent `var_coords`
/// externally.
#[derive(Clone, Debug)]
pub struct GlyphCacheKey {
/// Unique identifier for the font blob.
pub font_id: u64,
/// Index within font collection (for TTC files).
pub font_index: u32,
/// Glyph index within the font.
pub glyph_id: u32,
/// Font size as f32 bits (exact match, no quantization).
pub size_bits: u32,
/// Whether hinting was applied.
pub hinted: bool,
/// Horizontal subpixel position (0 to SUBPIXEL_BUCKETS-1 for outlines),
/// or a sentinel (`SUBPIXEL_COLR` / `SUBPIXEL_BITMAP`) for non-outline glyphs.
pub subpixel_x: u8,
/// Context color for COLR glyphs. Only used for rendering, not for Hash/Eq.
pub context_color: AlphaColor<Srgb>,
/// Pre-packed context color (premultiplied RGBA8 as u32) used in Hash/Eq.
pub context_color_packed: u32,
/// Synthetic embolden amount. Only non-zero for outline glyphs.
pub embolden_x_bits: u32,
/// Synthetic embolden amount. Only non-zero for outline glyphs.
pub embolden_y_bits: u32,
/// Join style for synthetic embolden. Only meaningful for outline glyphs.
pub embolden_join_bits: u8,
/// Miter limit for synthetic embolden. Only meaningful for outline glyphs.
pub embolden_miter_limit_bits: u32,
/// Tolerance for synthetic embolden. Only meaningful for outline glyphs.
pub embolden_tolerance_bits: u32,
/// Variation coordinates for variable fonts.
pub var_coords: SmallVec<[NormalizedCoord; 4]>,
}
impl GlyphCacheKey {
/// Creates a new cache key.
///
/// `fractional_x` (the fractional pixel offset) is quantized into
/// `SUBPIXEL_BUCKETS` buckets, so nearby positions share the same entry.
#[inline]
pub fn new(
font_id: u64,
font_index: u32,
glyph_id: u32,
size: f32,
hinted: bool,
fractional_x: f32,
context_color: AlphaColor<Srgb>,
context_color_packed: u32,
embolden: FontEmbolden,
var_coords: &[NormalizedCoord],
) -> Self {
Self {
font_id,
font_index,
glyph_id,
size_bits: size.to_bits(),
hinted,
subpixel_x: quantize_subpixel(fractional_x),
context_color,
context_color_packed,
embolden_x_bits: f32_bits(embolden.amount.xx),
embolden_y_bits: f32_bits(embolden.amount.yy),
embolden_join_bits: join_bits(embolden.join),
embolden_miter_limit_bits: f32_bits(embolden.miter_limit),
embolden_tolerance_bits: f32_bits(embolden.tolerance),
var_coords: SmallVec::from_slice(var_coords),
}
}
}
/// Manual `Hash` and `PartialEq` use the pre-packed `context_color_packed` field
/// (a premultiplied RGBA8 `u32`) instead of `AlphaColor<Srgb>`, which doesn't
/// implement `Hash`/`Eq`. Packing once at construction avoids repeated work
/// during lookups. `glyph_id` is compared first for early short-circuit.
impl Hash for GlyphCacheKey {
#[inline]
fn hash<H: Hasher>(&self, state: &mut H) {
self.font_id.hash(state);
self.font_index.hash(state);
self.glyph_id.hash(state);
self.size_bits.hash(state);
self.hinted.hash(state);
self.subpixel_x.hash(state);
self.context_color_packed.hash(state);
self.embolden_x_bits.hash(state);
self.embolden_y_bits.hash(state);
self.embolden_join_bits.hash(state);
self.embolden_miter_limit_bits.hash(state);
self.embolden_tolerance_bits.hash(state);
}
}
impl PartialEq for GlyphCacheKey {
#[inline]
fn eq(&self, other: &Self) -> bool {
self.glyph_id == other.glyph_id
&& self.subpixel_x == other.subpixel_x
&& self.font_id == other.font_id
&& self.font_index == other.font_index
&& self.size_bits == other.size_bits
&& self.hinted == other.hinted
&& self.context_color_packed == other.context_color_packed
&& self.embolden_x_bits == other.embolden_x_bits
&& self.embolden_y_bits == other.embolden_y_bits
&& self.embolden_join_bits == other.embolden_join_bits
&& self.embolden_miter_limit_bits == other.embolden_miter_limit_bits
&& self.embolden_tolerance_bits == other.embolden_tolerance_bits
}
}
impl Eq for GlyphCacheKey {}
#[inline(always)]
fn join_bits(join: Join) -> u8 {
match join {
Join::Bevel => 0,
Join::Miter => 1,
Join::Round => 2,
}
}
#[expect(
clippy::cast_possible_truncation,
reason = "Cache keys intentionally store embolden parameters at f32 precision."
)]
#[inline(always)]
fn f32_bits(value: f64) -> u32 {
(value as f32).to_bits()
}
/// Premultiply and pack an RGBA color into a `u32` for bitwise hashing/comparison.
#[inline]
pub(crate) fn pack_color(color: AlphaColor<Srgb>) -> u32 {
color.premultiply().to_rgba8().to_u32()
}
/// Quantize a fractional pixel offset into one of [`SUBPIXEL_BUCKETS`] buckets.
///
/// Values near 1.0 (>= 0.875 with 4 buckets) are clamped to the last bucket
/// rather than wrapping to 0. Wrapping to bucket 0 without also incrementing the
/// integer pixel coordinate would shift the glyph by ~0.75px in the wrong
/// direction. Clamping keeps the worst-case error to 0.125px.
#[expect(
clippy::cast_possible_truncation,
reason = "result is clamped to SUBPIXEL_BUCKETS-1 which fits in u8"
)]
#[inline]
fn quantize_subpixel(frac: f32) -> u8 {
let normalized = frac.fract();
let normalized = if normalized < 0.0 {
normalized + 1.0
} else {
normalized
};
((normalized * SUBPIXEL_BUCKETS as f32).round() as u8).min(SUBPIXEL_BUCKETS - 1)
}
/// Convert a quantized bucket index back to the fractional pixel offset it represents.
#[inline]
pub fn subpixel_offset(quantized: u8) -> f32 {
quantized as f32 / SUBPIXEL_BUCKETS as f32
}
#[cfg(test)]
mod tests {
use crate::color::palette::css::BLACK;
use super::*;
#[test]
fn test_quantize_subpixel() {
// Test bucket boundaries
assert_eq!(quantize_subpixel(0.0), 0);
assert_eq!(quantize_subpixel(0.1), 0);
assert_eq!(quantize_subpixel(0.2), 1);
assert_eq!(quantize_subpixel(0.25), 1);
assert_eq!(quantize_subpixel(0.4), 2);
assert_eq!(quantize_subpixel(0.5), 2);
assert_eq!(quantize_subpixel(0.6), 2);
assert_eq!(quantize_subpixel(0.7), 3);
assert_eq!(quantize_subpixel(0.75), 3);
assert_eq!(quantize_subpixel(0.9), 3);
assert_eq!(quantize_subpixel(1.0), 0);
}
#[test]
fn test_subpixel_offset() {
assert_eq!(subpixel_offset(0), 0.0);
assert_eq!(subpixel_offset(1), 0.25);
assert_eq!(subpixel_offset(2), 0.5);
assert_eq!(subpixel_offset(3), 0.75);
}
#[test]
fn test_key_equality() {
let packed = pack_color(BLACK);
let key1 = GlyphCacheKey::new(
1,
0,
42,
16.0,
true,
0.3,
BLACK,
packed,
FontEmbolden::default(),
&[],
);
let key2 = GlyphCacheKey::new(
1,
0,
42,
16.0,
true,
0.3,
BLACK,
packed,
FontEmbolden::default(),
&[],
);
assert_eq!(key1, key2);
}
#[test]
fn test_outline_colr_bitmap_keys_never_collide() {
let packed = pack_color(BLACK);
let outline_key = GlyphCacheKey::new(
1,
0,
42,
16.0,
false,
0.0,
BLACK,
packed,
FontEmbolden::default(),
&[],
);
let colr_key = GlyphCacheKey {
font_id: 1,
font_index: 0,
glyph_id: 42,
size_bits: 16.0_f32.to_bits(),
hinted: false,
subpixel_x: SUBPIXEL_COLR,
context_color: BLACK,
context_color_packed: packed,
embolden_x_bits: 0,
embolden_y_bits: 0,
embolden_join_bits: join_bits(Join::Miter),
embolden_miter_limit_bits: 4.0_f32.to_bits(),
embolden_tolerance_bits: 0.1_f32.to_bits(),
var_coords: SmallVec::new(),
};
let bitmap_key = GlyphCacheKey {
font_id: 1,
font_index: 0,
glyph_id: 42,
size_bits: 16.0_f32.to_bits(),
hinted: false,
subpixel_x: SUBPIXEL_BITMAP,
context_color: BLACK,
context_color_packed: packed,
embolden_x_bits: 0,
embolden_y_bits: 0,
embolden_join_bits: join_bits(Join::Miter),
embolden_miter_limit_bits: 4.0_f32.to_bits(),
embolden_tolerance_bits: 0.1_f32.to_bits(),
var_coords: SmallVec::new(),
};
assert_ne!(outline_key, colr_key);
assert_ne!(outline_key, bitmap_key);
assert_ne!(colr_key, bitmap_key);
}
#[test]
fn test_sentinels_unreachable_by_quantize() {
for i in 0..=255_u8 {
let frac = i as f32 / 255.0;
let bucket = quantize_subpixel(frac);
assert!(bucket < SUBPIXEL_BUCKETS, "bucket {bucket} for frac {frac}");
}
}
#[test]
fn test_var_coords_excluded_from_equality() {
let packed = pack_color(BLACK);
// var_coords is excluded from Hash/Eq (two-level map handles it),
// so keys differing only in var_coords are considered equal.
let key1 = GlyphCacheKey::new(
1,
0,
42,
16.0,
true,
0.3,
BLACK,
packed,
FontEmbolden::default(),
&[],
);
let key2 = GlyphCacheKey::new(
1,
0,
42,
16.0,
true,
0.3,
BLACK,
packed,
FontEmbolden::default(),
&[NormalizedCoord::from_bits(100)],
);
assert_eq!(key1, key2);
}
}