From e30c034738a2fb576d2a4370e305c1b1d0e21214 Mon Sep 17 00:00:00 2001 From: Ivan Smirnov Date: Mon, 17 Oct 2022 00:42:28 +0100 Subject: [PATCH 1/4] Bump bytemuck dependency --- Cargo.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index d1c8ec7..327a4d1 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -23,7 +23,7 @@ std = [] # std mode (enabled by default) - provides access to `std::io`, reference = [] # follows reference encoder implementation precisely, but may be slightly slower [dependencies] -bytemuck = "1.7" +bytemuck = "1.12" [workspace] members = ["libqoi", "bench"] From 0e799be72d4eb5957edf72a4b6deacdfb4ec2e5f Mon Sep 17 00:00:00 2001 From: Ivan Smirnov Date: Mon, 17 Oct 2022 00:42:55 +0100 Subject: [PATCH 2/4] Bump MSRV to 1.60, set edition to 2021 --- .github/workflows/ci.yml | 2 +- Cargo.toml | 4 ++-- README.md | 3 +-- 3 files changed, 4 insertions(+), 5 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 1839202..8f9f42a 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -13,7 +13,7 @@ jobs: runs-on: ubuntu-latest strategy: matrix: - rust: [stable, beta, nightly, 1.51.0] # MSRV=1.51 + rust: [stable, beta, nightly, 1.61.0] # MSRV=1.61 steps: - uses: actions/checkout@v2 with: {submodules: true} diff --git a/Cargo.toml b/Cargo.toml index 327a4d1..4300672 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -3,7 +3,7 @@ name = "qoi" version = "0.4.0" description = "VERY fast encoder/decoder for QOI (Quite Okay Image) format" authors = ["Ivan Smirnov "] -edition = "2018" +edition = "2021" readme = "README.md" license = "MIT/Apache-2.0" repository = "https://github.com/aldanor/qoi-rust" @@ -14,7 +14,7 @@ keywords = ["qoi", "graphics", "image", "encoding"] exclude = [ "assets/*", ] -rust-version = "1.51.0" +rust-version = "1.61.0" [features] default = ["std"] diff --git a/README.md b/README.md index a7de677..315d006 100644 --- a/README.md +++ b/README.md @@ -51,8 +51,7 @@ this library proved to be the fastest one by a noticeable margin. ### Rust version -The minimum required Rust version is 1.51.0 (any changes to this would be -considered to be a breaking change). +The minimum required Rust version for the latest crate version is 1.61.0. ### `no_std` From fe83d1b3eea80c83c0e4234b6a15ef237b8123ff Mon Sep 17 00:00:00 2001 From: Ivan Smirnov Date: Mon, 17 Oct 2022 00:51:38 +0100 Subject: [PATCH 3/4] Version bump to v0.4.1 --- Cargo.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Cargo.toml b/Cargo.toml index 4300672..8417cc7 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -1,6 +1,6 @@ [package] name = "qoi" -version = "0.4.0" +version = "0.4.1" description = "VERY fast encoder/decoder for QOI (Quite Okay Image) format" authors = ["Ivan Smirnov "] edition = "2021" From 8099e065f39322ec7b63a76accbb7044fad507d0 Mon Sep 17 00:00:00 2001 From: ZettaScript Date: Fri, 16 Dec 2022 22:12:43 +0100 Subject: [PATCH 4/4] wip --- README.md | 10 ++++++++++ src/consts.rs | 1 + src/decode.rs | 18 +++++++++++++++--- src/encode.rs | 31 +++++++++++++++++++++++-------- 4 files changed, 49 insertions(+), 11 deletions(-) diff --git a/README.md b/README.md index 315d006..d0a9651 100644 --- a/README.md +++ b/README.md @@ -16,6 +16,16 @@ Fast encoder/decoder for [QOI image format](https://qoiformat.org/), implemented - `no_std` support. - Roundtrip-tested vs the reference C implementation; fuzz-tested. +### Note about this fork + +This fork implements a slight improvement to the original specs, which leaves unused the `QOI_OP_RGBA` chunk flag with RGB. + +Here, we use this flag for the new `QOI_OP_RUN2` chunk. It's like the `QOI_OP_RUN` chunk, but followed by two bytes representing `run` (BE). (only for RGB, as the flag is already used for RGBA) + +The decoder remains fully compatible with the original one (except when using `QOI_OP_RGBA` in a RGB image). The encoder is fully compatible for RGBA, not for RGB (except using the `reference` feature). + +Why this? Because it enables significant improvements for compressing images with large uniform areas (such as screen captures), or for encoding a diff-filtered video stream where successive frames often have identical regions. (see [syeve](https://framagit.org/ZettaScript/syeve) for the video encoding) + ### Examples ```rust diff --git a/src/consts.rs b/src/consts.rs index 2281c24..0e39c4b 100644 --- a/src/consts.rs +++ b/src/consts.rs @@ -4,6 +4,7 @@ pub const QOI_OP_LUMA: u8 = 0x80; // 10xxxxxx pub const QOI_OP_RUN: u8 = 0xc0; // 11xxxxxx pub const QOI_OP_RGB: u8 = 0xfe; // 11111110 pub const QOI_OP_RGBA: u8 = 0xff; // 11111111 +pub const QOI_OP_RUN2: u8 = 0xff; // 11111111 pub const QOI_MASK_2: u8 = 0xc0; // (11)000000 diff --git a/src/decode.rs b/src/decode.rs index 019dac2..a4343d0 100644 --- a/src/decode.rs +++ b/src/decode.rs @@ -49,9 +49,21 @@ where px.update_rgb(*r, *g, *b); data = dtail; } - [QOI_OP_RGBA, r, g, b, a, dtail @ ..] if RGBA => { - px.update_rgba(*r, *g, *b, *a); - data = dtail; + [QOI_OP_RGBA, dtail @ ..] => { + if RGBA { + if let [r, g, b, a, dtail @ ..] = dtail { + px.update_rgba(*r, *g, *b, *a); + data = dtail; + } + } else if let [b1, b2, dtail @ ..] = dtail { + *px_out = px.into(); + let run = (u16::from_be_bytes([*b1, *b2]) as usize).min(pixels.len()); + let (phead, ptail) = pixels.split_at_mut(run); // can't panic + phead.fill(px.into()); + pixels = ptail; + data = dtail; + continue; + } } [b1 @ QOI_OP_RUN..=QOI_OP_RUN_END, dtail @ ..] => { *px_out = px.into(); diff --git a/src/encode.rs b/src/encode.rs index 0ed8476..ce2b36f 100644 --- a/src/encode.rs +++ b/src/encode.rs @@ -6,7 +6,7 @@ use std::io::Write; use bytemuck::Pod; -use crate::consts::{QOI_HEADER_SIZE, QOI_OP_INDEX, QOI_OP_RUN, QOI_PADDING, QOI_PADDING_SIZE}; +use crate::consts::{QOI_HEADER_SIZE, QOI_OP_INDEX, QOI_OP_RUN, QOI_PADDING, QOI_PADDING_SIZE, QOI_OP_RUN2}; use crate::error::{Error, Result}; use crate::header::Header; use crate::pixel::{Pixel, SupportedChannels}; @@ -26,7 +26,7 @@ where let mut index = [Pixel::new(); 256]; let mut px_prev = Pixel::new().with_a(0xff); let mut hash_prev = px_prev.hash_index(); - let mut run = 0_u8; + let mut run = 0_u16; let mut px = Pixel::::new().with_a(0xff); let mut index_allowed = false; @@ -36,8 +36,20 @@ where px.read(chunk); if px == px_prev { run += 1; + #[cfg(not(feature = "reference"))] + if N == 3 { + if run == 65535 || unlikely(i == n_pixels - 1) { + let [b1, b2] = (run-1).to_be_bytes(); + buf = buf.write_many(&[QOI_OP_RUN2, b1, b2])?; + run = 0; + } + } else if run == 62 || unlikely(i == n_pixels - 1) { + buf = buf.write_one(QOI_OP_RUN | (run as u8 - 1))?; + run = 0; + } + #[cfg(feature = "reference")] if run == 62 || unlikely(i == n_pixels - 1) { - buf = buf.write_one(QOI_OP_RUN | (run - 1))?; + buf = buf.write_one(QOI_OP_RUN | (run as u8 - 1))?; run = 0; } } else { @@ -45,15 +57,18 @@ where #[cfg(not(feature = "reference"))] { // credits for the original idea: @zakarumych (had to be fixed though) - buf = buf.write_one(if run == 1 && index_allowed { - QOI_OP_INDEX | hash_prev + buf = if run == 1 && index_allowed { + buf.write_one(QOI_OP_INDEX | hash_prev)? + } else if N == 4 || run < 63 { + buf.write_one(QOI_OP_RUN | (run as u8 - 1))? } else { - QOI_OP_RUN | (run - 1) - })?; + let [b1, b2] = (run-1).to_be_bytes(); + buf.write_many(&[QOI_OP_RUN2, b1, b2])? + }; } #[cfg(feature = "reference")] { - buf = buf.write_one(QOI_OP_RUN | (run - 1))?; + buf = buf.write_one(QOI_OP_RUN | (run as u8 - 1))?; } run = 0; }