From 337fca5acec1fb94747aa93a051287d5efa27b4d Mon Sep 17 00:00:00 2001 From: I <1091761+wx257osn2@users.noreply.github.com> Date: Wed, 21 Feb 2024 12:04:05 +0900 Subject: [PATCH 1/3] add CI with official test suite --- .github/workflows/linux.yml | 42 +++++++++++++++++++++++++++++--- .github/workflows/mac.yml | 42 +++++++++++++++++++++++++++++--- .github/workflows/windows.yml | 46 +++++++++++++++++++++++++++++++---- 3 files changed, 117 insertions(+), 13 deletions(-) diff --git a/.github/workflows/linux.yml b/.github/workflows/linux.yml index 7348f88..b9205a0 100644 --- a/.github/workflows/linux.yml +++ b/.github/workflows/linux.yml @@ -29,11 +29,24 @@ jobs: - id: cached uses: andstor/file-existence-action@v3 with: - files: images - - name: get benchmark suite + files: "images, qoi_test_images" + - name: get test/benchmark suite if: steps.cached.outputs.files_exists == 'false' shell: bash - run: curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + run: | + curl -O https://qoiformat.org/qoi_test_images.zip + unzip qoi_test_images.zip + rm qoi_test_images.zip + curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + - name: build reference qoiconv + shell: bash + run: | + pushd .dependencies/qoi + ln -s ../stb/stb_image.h . + ln -s ../stb/stb_image_write.h . + make conv + popd + mv .dependencies/qoi/qoiconv bin/qoiconv_orig - name: build shell: bash run: CXX=clang++-17 make -j @@ -42,4 +55,25 @@ jobs: run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: bash - run: bin/test + run: | + pushd qoi_test_images + mkdir -p /tmp/qoixx + mkdir -p /tmp/qoi + for i in ./*.qoi; do + # test decode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.png + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.png + diff /tmp/qoi{,xx}/${i%.*}.png + done + rm -rf /tmp/qoixx/*.png + rm -rf /tmp/qoi/*.png + for i in ./*.png; do + # test encode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoixx/${i%.*}.qoi /tmp/qoixx/${i} + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoi/${i%.*}.qoi /tmp/qoi/${i} + diff /tmp/qoi{,xx}/${i} + done + popd + bin/test diff --git a/.github/workflows/mac.yml b/.github/workflows/mac.yml index 345a68f..a76936f 100644 --- a/.github/workflows/mac.yml +++ b/.github/workflows/mac.yml @@ -23,11 +23,24 @@ jobs: - id: cached uses: andstor/file-existence-action@v3 with: - files: images - - name: get benchmark suite + files: "images, qoi_test_images" + - name: get test/benchmark suite if: steps.cached.outputs.files_exists == 'false' shell: bash - run: curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + run: | + curl -O https://qoiformat.org/qoi_test_images.zip + unzip qoi_test_images.zip + rm qoi_test_images.zip + curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + - name: build reference qoiconv + shell: bash + run: | + pushd .dependencies/qoi + ln -s ../stb/stb_image.h . + ln -s ../stb/stb_image_write.h . + make conv + popd + mv .dependencies/qoi/qoiconv bin/qoiconv_orig - name: build shell: bash run: CXX=$(brew --prefix llvm@17)/bin/clang++ make -j @@ -36,4 +49,25 @@ jobs: run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: bash - run: bin/test + run: | + pushd qoi_test_images + mkdir -p /tmp/qoixx + mkdir -p /tmp/qoi + for i in ./*.qoi; do + # test decode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.png + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.png + diff /tmp/qoi{,xx}/${i%.*}.png + done + rm -rf /tmp/qoixx/*.png + rm -rf /tmp/qoi/*.png + for i in ./*.png; do + # test encode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoixx/${i%.*}.qoi /tmp/qoixx/${i} + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoi/${i%.*}.qoi /tmp/qoi/${i} + diff /tmp/qoi{,xx}/${i} + done + popd + bin/test diff --git a/.github/workflows/windows.yml b/.github/workflows/windows.yml index 2399cad..d97d4ff 100644 --- a/.github/workflows/windows.yml +++ b/.github/workflows/windows.yml @@ -17,6 +17,8 @@ jobs: install: >- git curl + unzip + diffutils mingw-w64-ucrt-x86_64-make mingw-w64-ucrt-x86_64-clang - uses: actions/checkout@v4 @@ -29,11 +31,24 @@ jobs: - id: cached uses: andstor/file-existence-action@v3 with: - files: images - - name: get benchmark suite + files: "images, qoi_test_images" + - name: get test/benchmark suite if: steps.cached.outputs.files_exists == 'false' - shell: bash - run: curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + shell: msys2 {0} + run: | + curl -O https://qoiformat.org/qoi_test_images.zip + unzip qoi_test_images.zip + rm qoi_test_images.zip + curl https://qoiformat.org/benchmark/qoi_benchmark_suite.tar | tar x + - name: build reference qoiconv + shell: msys2 {0} + run: | + pushd .dependencies/qoi + ln -s ../stb/stb_image.h . + ln -s ../stb/stb_image_write.h . + mingw32-make conv + popd + mv .dependencies/qoi/qoiconv.exe bin/qoiconv_orig.exe - name: build shell: msys2 {0} run: CXX=clang++ mingw32-make -j @@ -42,4 +57,25 @@ jobs: run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: msys2 {0} - run: bin/test + run: | + pushd qoi_test_images + mkdir -p /tmp/qoixx + mkdir -p /tmp/qoi + for i in ./*.qoi; do + # test decode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.png + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.png + diff /tmp/qoi{,xx}/${i%.*}.png + done + rm -rf /tmp/qoixx/*.png + rm -rf /tmp/qoi/*.png + for i in ./*.png; do + # test encode + ../bin/qoiconv ${i} /tmp/qoixx/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoixx/${i%.*}.qoi /tmp/qoixx/${i} + ../bin/qoiconv_orig ${i} /tmp/qoi/${i%.*}.qoi + ../bin/qoiconv_orig /tmp/qoi/${i%.*}.qoi /tmp/qoi/${i} + diff /tmp/qoi{,xx}/${i} + done + popd + bin/test From 31ee62b59f9f395f16c95c505c5f3ed1889450b9 Mon Sep 17 00:00:00 2001 From: I <1091761+wx257osn2@users.noreply.github.com> Date: Sat, 24 Feb 2024 09:40:51 +0900 Subject: [PATCH 2/3] test before run --- .github/workflows/linux.yml | 6 +++--- .github/workflows/mac.yml | 6 +++--- .github/workflows/windows.yml | 6 +++--- 3 files changed, 9 insertions(+), 9 deletions(-) diff --git a/.github/workflows/linux.yml b/.github/workflows/linux.yml index b9205a0..212b59f 100644 --- a/.github/workflows/linux.yml +++ b/.github/workflows/linux.yml @@ -50,9 +50,6 @@ jobs: - name: build shell: bash run: CXX=clang++-17 make -j - - name: run - shell: bash - run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: bash run: | @@ -77,3 +74,6 @@ jobs: done popd bin/test + - name: run + shell: bash + run: bin/qoibench 1 images --noreference --nowarmup diff --git a/.github/workflows/mac.yml b/.github/workflows/mac.yml index a76936f..5abf18e 100644 --- a/.github/workflows/mac.yml +++ b/.github/workflows/mac.yml @@ -44,9 +44,6 @@ jobs: - name: build shell: bash run: CXX=$(brew --prefix llvm@17)/bin/clang++ make -j - - name: run - shell: bash - run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: bash run: | @@ -71,3 +68,6 @@ jobs: done popd bin/test + - name: run + shell: bash + run: bin/qoibench 1 images --noreference --nowarmup diff --git a/.github/workflows/windows.yml b/.github/workflows/windows.yml index d97d4ff..00417bf 100644 --- a/.github/workflows/windows.yml +++ b/.github/workflows/windows.yml @@ -52,9 +52,6 @@ jobs: - name: build shell: msys2 {0} run: CXX=clang++ mingw32-make -j - - name: run - shell: msys2 {0} - run: bin/qoibench 1 images --noreference --nowarmup - name: test shell: msys2 {0} run: | @@ -79,3 +76,6 @@ jobs: done popd bin/test + - name: run + shell: msys2 {0} + run: bin/qoibench 1 images --noreference --nowarmup From 9c3a2540dd4049c2cff65fc641b16651d95c5b1e Mon Sep 17 00:00:00 2001 From: I <1091761+wx257osn2@users.noreply.github.com> Date: Wed, 21 Feb 2024 15:34:33 +0900 Subject: [PATCH 3/3] fix edge case in decoder when very first op is QOI_OP_RUN --- include/qoixx.hpp | 13 +++++++++++-- 1 file changed, 11 insertions(+), 2 deletions(-) diff --git a/include/qoixx.hpp b/include/qoixx.hpp index 54c4871..62ebbfe 100644 --- a/include/qoixx.hpp +++ b/include/qoixx.hpp @@ -11,6 +11,7 @@ #include #include #include +#include #ifndef QOIXX_NO_SIMD #if defined(__ARM_FEATURE_SVE) @@ -1146,7 +1147,7 @@ class qoi{ static constexpr auto hash_diff_table = luma_hash_diff_table.data() + hash_table_offset; ) - const auto f = [&pixels, &p, &px_len, &size, &px, &index QOIXX_HPP_WITH_TABLES(, &hash)]{ + const auto f = [&pixels, &p, &px_len, &size, &px, &index QOIXX_HPP_WITH_TABLES(, &hash)](bool first){ static constexpr std::uint32_t mask_tail_6 = 0b0011'1111u; [[maybe_unused]] static constexpr std::uint32_t mask_tail_4 = 0b0000'1111u; [[maybe_unused]] static constexpr std::uint32_t mask_tail_2 = 0b0000'0011u; @@ -1187,6 +1188,13 @@ class qoi{ run = px_len; px_len -= run; QOIXX_HPP_DECODE_RUN(px, run) + if(first)[[unlikely]]{ + QOIXX_HPP_WITH_TABLES(hash = (0*3+0*5+0*7+255*11) % index_size;) + if constexpr(std::is_same::value) + index[QOIXX_HPP_WITH_TABLES(hash) QOIXX_HPP_WITHOUT_TABLES((0*3+0*5+0*7+255*11) % index_size)] = px; + else + efficient_memcpy(index + QOIXX_HPP_WITH_TABLES(hash) QOIXX_HPP_WITHOUT_TABLES((0*3+0*5+0*7+255*11) % index_size), &px); + } return; } if(b1 == chunk_tag::rgb){ @@ -1271,8 +1279,9 @@ class qoi{ push(pixels, &px); }; + bool first = true; while(px_len--)[[likely]]{ - f(); + f(std::exchange(first, false)); if(size < sizeof(padding))[[unlikely]]{ throw std::runtime_error("qoixx::qoi::decode: insufficient input data"); }