garb
Dress your pixels for the occasion.
You can't show up to a function in the wrong style. Get with the times and swap your BGR for your RGB, your ARGB for your RGBA, your BGRX for your BGRA, and tie up loose ends like that unreliable alpha.
SIMD-accelerated pixel format conversions: x86-64 AVX2, ARM NEON, WASM SIMD128,
with automatic scalar fallback. no_std compatible, #![forbid(unsafe_code)].
What it does
Converts between pixel layouts at the byte-slice level. Every image decoder and renderer has an opinion about channel order and pixel width, and none of them agree. garb handles the mechanical part — swapping, expanding, and stripping channels — so you can get back to the interesting work.
SIMD-optimized (contiguous and strided)
- RGBA ↔ BGRA (in-place and copy)
- RGB ↔ BGR (in-place and copy)
- RGB → RGBA / BGRA
- BGR → BGRA / RGBA
- RGBA / BGRA → RGB / BGR (drop alpha)
- Gray → RGBA
- GrayAlpha → RGBA
- Fill alpha (set byte 3 = 255 in each 4-byte pixel, for RGBA/BGRA layouts)
All operations have _strided variants for images with padding between rows
(common in video frames and GPU textures).
Performance
Benchmarked on 1920×1080 buffers (x86-64 AVX2, Zen 4). "Naive" is the
obvious chunks_exact + swap/copy loop — what the compiler autovectorizes
on its own.
| Operation | garb | naive | speedup | throughput |
|---|---|---|---|---|
| RGBA ↔ BGRA (in-place) | 81 µs | 644 µs | 8.0x | 96 GiB/s |
| RGB ↔ BGR (in-place) | 208 µs | 823 µs | 4.0x | 28 GiB/s |
| RGB ↔ BGR (copy) | 128 µs | 1,059 µs | 8.3x | 45 GiB/s |
| RGBA → RGB (strip alpha) | 147 µs | 857 µs | 5.8x | 52 GiB/s |
| BGRA → RGB (strip + swap) | 147 µs | 859 µs | 5.8x | 52 GiB/s |
| RGB → RGBA (expand) | 146 µs | 1,056 µs | 7.2x | 53 GiB/s |
| Fill alpha | 84 µs | 206 µs | 2.5x | 92 GiB/s |
Run cargo bench to reproduce.
Usage
The core &[u8] / &mut [u8] API lives in garb::bytes.
Every function returns Result<(), SizeError> — no panics, no silent truncation.
use ;
use SizeError;
// In-place: swap R↔B in a 4bpp buffer
let mut pixels = vec!;
rgba_to_bgra_inplace?;
assert_eq!;
// Copy: RGB (3 bpp) → BGRA (4 bpp), alpha filled to 255
let rgb = vec!;
let mut bgra = vec!;
rgb_to_bgra?;
assert_eq!;
# Ok::
Strided images
A stride is the distance between the start of one row and the start of
the next, measured in units of the slice's element type. For the core &[u8]
API that means bytes; for the typed imgref API it means elements of the
slice's item type (e.g. pixel count for ImgRef<Rgba<u8>>). When
stride > width the gap is padding — garb never reads or writes it.
All _strided functions take dimensions before strides:
- In-place:
(buf, width, height, stride) - Copy:
(src, dst, width, height, src_stride, dst_stride)
use ;
// In-place: 60 pixels wide, stride=256 bytes, 100 rows
let mut buf = vec!;
rgba_to_bgra_inplace_strided?;
// Copy with different strides: RGB (stride=192) → BGRA (stride=256)
let rgb_buf = vec!;
let mut bgra_buf = vec!;
rgb_to_bgra_strided?;
# Ok::
Type-safe conversions (feature rgb)
If you're already using the rgb crate,
garb::typed_rgb gives you typed wrappers. In-place swaps return
reinterpreted references — zero-copy, zero-allocation.
use ;
use typed_rgb;
let mut pixels: = vec!;
let bgra: &mut = rgba_to_bgra_mut;
Whole-image conversions (feature imgref)
garb::imgref handles strided ImgVec / ImgRef / ImgRefMut types
from the imgref crate. In-place swaps
consume and return the ImgVec with the buffer reinterpreted. Copy
conversions take ImgRef + ImgRefMut — you own the destination buffer.
use ;
use ImgVec;
use imgref;
let rgba_img = new;
let bgra_img: = swap_rgba_to_bgra;
Feature flags
| Feature | Default | What it adds |
|---|---|---|
std |
yes | Enables std on dependencies (e.g. archmage) |
rgb |
no | garb::typed_rgb — conversions on Rgba<u8>, Bgra<u8>, etc. |
imgref |
no | garb::imgref — whole-image conversions on ImgVec / ImgRef (implies rgb) |
The crate is no_std + alloc by default. The imgref feature requires alloc.
The core byte-slice API has no allocator requirement.
SIMD dispatch
garb uses archmage for runtime SIMD detection with compile-time acceleration. On each platform:
- x86-64: AVX2 (checked at runtime via
cpuid) - aarch64: NEON (compile-time guaranteed on AArch64)
- wasm32: SIMD128 (compile-time via
target-feature=+simd128) - Fallback: Scalar code on all platforms, always available
The first call to each function detects and caches the best available tier.
There's no setup, no feature flags to configure, and no unsafe — archmage
handles it all behind safe token types.
Function reference
Functions follow {src}_to_{dst} for copies, {src}_to_{dst}_inplace for
mutations. Symmetric swaps (like RGBA↔BGRA) provide both names as aliases.
Append _strided for padded row layouts.
Core API — garb::bytes (&[u8])
Every function returns Result<(), SizeError>. All have _strided variants.
| Function | Operation |
|---|---|
rgba_to_bgra_inplace |
Swap R↔B in 4bpp buffer (RGBA↔BGRA) |
rgba_to_bgra |
Copy 4bpp, swapping R↔B |
rgb_to_bgr_inplace |
Swap R↔B in 3bpp buffer (RGB↔BGR) |
rgb_to_bgr |
Copy 3bpp, swapping R↔B |
rgb_to_rgba |
3bpp → 4bpp, alpha = 255 |
rgb_to_bgra |
3bpp → 4bpp, swap R↔B, alpha = 255 |
bgr_to_rgba |
3bpp → 4bpp, swap R↔B, alpha = 255 |
bgr_to_bgra |
3bpp → 4bpp, alpha = 255 |
rgba_to_rgb |
4bpp → 3bpp, drop alpha |
bgra_to_rgb |
4bpp → 3bpp, swap R↔B, drop alpha |
bgra_to_bgr |
4bpp → 3bpp, drop alpha |
rgba_to_bgr |
4bpp → 3bpp, swap R↔B, drop alpha |
gray_to_rgba |
1bpp → 4bpp (R=G=B=gray, A=255) |
gray_alpha_to_rgba |
2bpp → 4bpp (R=G=B=gray, A=alpha) |
fill_alpha_rgba |
Set byte 3 to 255 in each 4-byte pixel (alpha-last: RGBA/BGRA) |
Aliases: bgra_to_rgba_inplace, bgra_to_rgba, bgr_to_rgb_inplace,
bgr_to_rgb, gray_to_bgra, gray_alpha_to_bgra (same underlying
operations).
typed_rgb (feature rgb)
Typed wrappers using rgb crate pixel types. In-place swaps return
reinterpreted &mut references (zero-copy).
| Function | Signature |
|---|---|
rgba_to_bgra_mut |
&mut [Rgba<u8>] → &mut [Bgra<u8>] |
bgra_to_rgba_mut |
&mut [Bgra<u8>] → &mut [Rgba<u8>] |
rgb_to_bgr_mut |
&mut [Rgb<u8>] → &mut [Bgr<u8>] |
bgr_to_rgb_mut |
&mut [Bgr<u8>] → &mut [Rgb<u8>] |
rgba_to_bgra_buf |
&[Rgba<u8>] + &mut [Bgra<u8>] |
bgra_to_rgba_buf |
&[Bgra<u8>] + &mut [Rgba<u8>] |
rgb_to_bgra_buf |
&[Rgb<u8>] + &mut [Bgra<u8>] |
rgb_to_rgba_buf |
&[Rgb<u8>] + &mut [Rgba<u8>] |
bgr_to_rgba_buf |
&[Bgr<u8>] + &mut [Rgba<u8>] |
bgr_to_bgra_buf |
&[Bgr<u8>] + &mut [Bgra<u8>] |
gray_to_rgba_buf |
&[Gray<u8>] + &mut [Rgba<u8>] |
gray_to_bgra_buf |
&[Gray<u8>] + &mut [Bgra<u8>] |
gray_alpha_to_rgba_buf |
&[GrayAlpha<u8>] + &mut [Rgba<u8>] |
gray_alpha_to_bgra_buf |
&[GrayAlpha<u8>] + &mut [Bgra<u8>] |
rgba_to_rgb_buf |
&[Rgba<u8>] + &mut [Rgb<u8>] |
bgra_to_bgr_buf |
&[Bgra<u8>] + &mut [Bgr<u8>] |
bgra_to_rgb_buf |
&[Bgra<u8>] + &mut [Rgb<u8>] |
rgba_to_bgr_buf |
&[Rgba<u8>] + &mut [Bgr<u8>] |
fill_alpha_rgba |
Set A=255 in &mut [Rgba<u8>] |
fill_alpha_bgra |
Set A=255 in &mut [Bgra<u8>] |
imgref (feature imgref)
Whole-image conversions on ImgVec / ImgRef / ImgRefMut. In-place
swaps consume and return an ImgVec with the buffer reinterpreted.
| Function | Operation |
|---|---|
swap_rgba_to_bgra |
ImgVec<Rgba<u8>> → ImgVec<Bgra<u8>> |
swap_bgra_to_rgba |
ImgVec<Bgra<u8>> → ImgVec<Rgba<u8>> |
convert_rgba_to_bgra |
ImgRef<Rgba<u8>> + ImgRefMut<Bgra<u8>> |
convert_bgra_to_rgba |
ImgRef<Bgra<u8>> + ImgRefMut<Rgba<u8>> |
convert_rgb_to_bgra |
ImgRef<Rgb<u8>> + ImgRefMut<Bgra<u8>> |
convert_rgb_to_rgba |
ImgRef<Rgb<u8>> + ImgRefMut<Rgba<u8>> |
convert_bgr_to_rgba |
ImgRef<Bgr<u8>> + ImgRefMut<Rgba<u8>> |
convert_bgr_to_bgra |
ImgRef<Bgr<u8>> + ImgRefMut<Bgra<u8>> |
convert_gray_to_rgba |
ImgRef<Gray<u8>> + ImgRefMut<Rgba<u8>> |
convert_gray_to_bgra |
ImgRef<Gray<u8>> + ImgRefMut<Bgra<u8>> |
convert_gray_alpha_to_rgba |
ImgRef<GrayAlpha<u8>> + ImgRefMut<Rgba<u8>> |
convert_gray_alpha_to_bgra |
ImgRef<GrayAlpha<u8>> + ImgRefMut<Bgra<u8>> |
convert_rgba_to_rgb |
ImgRef<Rgba<u8>> + ImgRefMut<Rgb<u8>> |
convert_bgra_to_rgb |
ImgRef<Bgra<u8>> + ImgRefMut<Rgb<u8>> |
convert_bgra_to_bgr |
ImgRef<Bgra<u8>> + ImgRefMut<Bgr<u8>> |
convert_rgba_to_bgr |
ImgRef<Rgba<u8>> + ImgRefMut<Bgr<u8>> |
License
MIT OR Apache-2.0