2021-02-01 09:03:04 +00:00
/*
* Copyright ( c ) 2021 , the SerenityOS developers .
2021-04-29 01:38:16 +00:00
* Copyright ( c ) 2021 , Brian Gianforcaro < bgianf @ serenityos . org >
2021-02-01 09:03:04 +00:00
*
2021-04-22 08:24:48 +00:00
* SPDX - License - Identifier : BSD - 2 - Clause
2021-02-01 09:03:04 +00:00
*/
2022-12-04 18:02:33 +00:00
# include <AK/DeprecatedString.h>
2021-11-23 10:32:25 +00:00
# include <LibCore/MappedFile.h>
2023-03-21 18:58:06 +00:00
# include <LibGfx/ImageFormats/BMPLoader.h>
2023-10-06 17:51:57 +00:00
# include <LibGfx/ImageFormats/DDSLoader.h>
2023-03-21 18:58:06 +00:00
# include <LibGfx/ImageFormats/GIFLoader.h>
# include <LibGfx/ImageFormats/ICOLoader.h>
2023-08-03 08:52:44 +00:00
# include <LibGfx/ImageFormats/ILBMLoader.h>
2023-03-21 18:58:06 +00:00
# include <LibGfx/ImageFormats/ImageDecoder.h>
# include <LibGfx/ImageFormats/JPEGLoader.h>
2023-07-22 03:04:29 +00:00
# include <LibGfx/ImageFormats/JPEGXLLoader.h>
2023-03-21 18:58:06 +00:00
# include <LibGfx/ImageFormats/PBMLoader.h>
# include <LibGfx/ImageFormats/PGMLoader.h>
# include <LibGfx/ImageFormats/PNGLoader.h>
# include <LibGfx/ImageFormats/PPMLoader.h>
# include <LibGfx/ImageFormats/TGALoader.h>
2023-07-02 22:09:27 +00:00
# include <LibGfx/ImageFormats/TinyVGLoader.h>
2023-03-21 18:58:06 +00:00
# include <LibGfx/ImageFormats/WebPLoader.h>
2021-04-29 01:38:16 +00:00
# include <LibTest/TestCase.h>
2021-02-01 09:03:04 +00:00
# include <stdio.h>
# include <string.h>
2023-01-31 19:00:33 +00:00
# ifdef AK_OS_SERENITY
# define TEST_INPUT(x) (" / usr / Tests / LibGfx / test-inputs / " x)
# else
# define TEST_INPUT(x) ("test-inputs / " x)
# endif
2023-06-20 00:46:43 +00:00
static Gfx : : ImageFrameDescriptor expect_single_frame ( Gfx : : ImageDecoderPlugin & plugin_decoder )
{
EXPECT_EQ ( plugin_decoder . frame_count ( ) , 1u ) ;
EXPECT ( ! plugin_decoder . is_animated ( ) ) ;
EXPECT ( ! plugin_decoder . loop_count ( ) ) ;
auto frame = MUST ( plugin_decoder . frame ( 0 ) ) ;
EXPECT_EQ ( frame . duration , 0 ) ;
return frame ;
}
static Gfx : : ImageFrameDescriptor expect_single_frame_of_size ( Gfx : : ImageDecoderPlugin & plugin_decoder , Gfx : : IntSize size )
{
EXPECT_EQ ( plugin_decoder . size ( ) , size ) ;
2023-07-07 03:55:49 +00:00
auto frame = expect_single_frame ( plugin_decoder ) ;
2023-06-20 00:46:43 +00:00
EXPECT_EQ ( frame . image - > size ( ) , size ) ;
return frame ;
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_bmp )
2021-02-01 09:03:04 +00:00
{
2023-10-18 18:37:35 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " bmp/rgba32-1.bmp " sv ) ) ) ;
EXPECT ( Gfx : : BMPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : BMPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame ( * plugin_decoder ) ;
}
TEST_CASE ( test_bmp_top_down )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " bmp/top-down.bmp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : BMPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : BMPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2021-02-01 09:03:04 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2023-10-21 22:22:32 +00:00
TEST_CASE ( test_ico_malformed_frame )
{
Array test_inputs = {
TEST_INPUT ( " ico/oss-fuzz-testcase-62541.ico " sv ) ,
2023-10-19 21:10:53 +00:00
TEST_INPUT ( " ico/oss-fuzz-testcase-63177.ico " sv ) ,
TEST_INPUT ( " ico/oss-fuzz-testcase-63357.ico " sv )
2023-10-21 22:22:32 +00:00
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder = TRY_OR_FAIL ( Gfx : : ICOImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
auto frame_or_error = plugin_decoder - > frame ( 0 ) ;
EXPECT ( frame_or_error . is_error ( ) ) ;
}
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_gif )
2021-02-01 09:03:04 +00:00
{
2023-01-31 19:00:33 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " download-animation.gif " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : GIFImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : GIFImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2021-02-01 09:03:04 +00:00
2023-01-20 08:13:14 +00:00
EXPECT ( plugin_decoder - > frame_count ( ) ) ;
2023-01-20 16:41:21 +00:00
EXPECT ( plugin_decoder - > is_animated ( ) ) ;
2023-01-20 08:13:14 +00:00
EXPECT ( ! plugin_decoder - > loop_count ( ) ) ;
2021-02-01 09:03:04 +00:00
2023-02-25 23:58:19 +00:00
auto frame = MUST ( plugin_decoder - > frame ( 1 ) ) ;
2021-11-11 10:20:58 +00:00
EXPECT ( frame . duration = = 400 ) ;
2021-02-01 09:03:04 +00:00
}
2022-12-18 18:13:19 +00:00
TEST_CASE ( test_not_ico )
2021-02-01 09:03:04 +00:00
{
2023-10-21 21:26:44 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " png/buggie.png " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( ! Gfx : : ICOImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-07-17 16:24:16 +00:00
EXPECT ( Gfx : : ICOImageDecoderPlugin : : create ( file - > bytes ( ) ) . is_error ( ) ) ;
2021-02-01 09:03:04 +00:00
}
2022-12-18 18:13:19 +00:00
TEST_CASE ( test_bmp_embedded_in_ico )
{
2023-01-31 19:00:33 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " serenity.ico " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : ICOImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : ICOImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2022-12-18 18:13:19 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2022-12-18 18:13:19 +00:00
}
2023-08-03 08:52:44 +00:00
TEST_CASE ( test_ilbm )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " ilbm/gradient.iff " sv ) ) ) ;
EXPECT ( Gfx : : ILBMImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : ILBMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-09-11 08:46:15 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 320 , 200 } ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 8 , 0 ) , Gfx : : Color ( 0xee , 0xbb , 0 , 255 ) ) ;
}
TEST_CASE ( test_ilbm_uncompressed )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " ilbm/gradient-uncompressed.iff " sv ) ) ) ;
EXPECT ( Gfx : : ILBMImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : ILBMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 320 , 200 } ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 8 , 0 ) , Gfx : : Color ( 0xee , 0xbb , 0 , 255 ) ) ;
2023-08-03 08:52:44 +00:00
}
2023-10-21 22:22:32 +00:00
TEST_CASE ( test_ilbm_malformed_header )
{
Array test_inputs = {
TEST_INPUT ( " ilbm/oss-fuzz-testcase-62033.iff " sv ) ,
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder_or_error = Gfx : : ILBMImageDecoderPlugin : : create ( file - > bytes ( ) ) ;
EXPECT ( plugin_decoder_or_error . is_error ( ) ) ;
}
}
TEST_CASE ( test_ilbm_malformed_frame )
{
Array test_inputs = {
TEST_INPUT ( " ilbm/oss-fuzz-testcase-63296.iff " sv )
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder = TRY_OR_FAIL ( Gfx : : ILBMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
auto frame_or_error = plugin_decoder - > frame ( 0 ) ;
EXPECT ( frame_or_error . is_error ( ) ) ;
}
}
2023-02-26 22:02:07 +00:00
TEST_CASE ( test_jpeg_sof0_one_scan )
2021-02-01 09:03:04 +00:00
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/rgb24.jpg " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-18 21:09:16 +00:00
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2021-02-01 09:03:04 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2023-02-26 22:07:14 +00:00
TEST_CASE ( test_jpeg_sof0_several_scans )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/several_scans.jpg " sv ) ) ) ;
2023-02-26 22:07:14 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 592 , 800 } ) ;
2023-02-26 22:07:14 +00:00
}
2023-03-05 03:05:02 +00:00
TEST_CASE ( test_jpeg_rgb_components )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/rgb_components.jpg " sv ) ) ) ;
2023-03-05 03:05:02 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 592 , 800 } ) ;
2023-03-05 03:05:02 +00:00
}
2023-02-27 19:32:41 +00:00
TEST_CASE ( test_jpeg_sof2_spectral_selection )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/spectral_selection.jpg " sv ) ) ) ;
2023-02-27 19:32:41 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 592 , 800 } ) ;
2023-02-27 19:32:41 +00:00
}
2023-03-04 04:45:47 +00:00
TEST_CASE ( test_jpeg_sof0_several_scans_odd_number_mcu )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/several_scans_odd_number_mcu.jpg " sv ) ) ) ;
2023-03-04 04:45:47 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 600 , 600 } ) ;
2023-03-04 04:45:47 +00:00
}
2023-03-18 03:39:30 +00:00
TEST_CASE ( test_jpeg_sof2_successive_aproximation )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/successive_approximation.jpg " sv ) ) ) ;
2023-03-18 03:39:30 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 600 , 800 } ) ;
2023-03-18 03:39:30 +00:00
}
2023-05-07 20:35:54 +00:00
TEST_CASE ( test_jpeg_sof1_12bits )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/12-bit.jpg " sv ) ) ) ;
2023-05-07 20:35:54 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 320 , 240 } ) ;
2023-05-07 20:35:54 +00:00
}
TEST_CASE ( test_jpeg_sof2_12bits )
{
2023-06-18 22:55:38 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/12-bit-progressive.jpg " sv ) ) ) ;
2023-05-07 20:35:54 +00:00
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 320 , 240 } ) ;
2023-05-07 20:35:54 +00:00
}
2023-07-05 15:08:35 +00:00
TEST_CASE ( test_jpeg_empty_icc )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/gradient_empty_icc.jpg " sv ) ) ) ;
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-07-05 15:19:43 +00:00
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 80 , 80 } ) ;
}
TEST_CASE ( test_jpeg_grayscale_with_app14 )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jpg/grayscale_app14.jpg " sv ) ) ) ;
EXPECT ( Gfx : : JPEGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-07-05 15:08:35 +00:00
auto plugin_decoder = MUST ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 80 , 80 } ) ;
}
2023-10-21 22:22:32 +00:00
TEST_CASE ( test_jpeg_malformed_header )
{
Array test_inputs = {
TEST_INPUT ( " jpg/oss-fuzz-testcase-59785.jpg " sv )
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder_or_error = Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ;
EXPECT ( plugin_decoder_or_error . is_error ( ) ) ;
}
}
TEST_CASE ( test_jpeg_malformed_frame )
{
Array test_inputs = {
2023-11-03 20:29:31 +00:00
TEST_INPUT ( " jpg/oss-fuzz-testcase-62584.jpg " sv ) ,
TEST_INPUT ( " jpg/oss-fuzz-testcase-63815.jpg " sv )
2023-10-21 22:22:32 +00:00
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder = TRY_OR_FAIL ( Gfx : : JPEGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
auto frame_or_error = plugin_decoder - > frame ( 0 ) ;
EXPECT ( frame_or_error . is_error ( ) ) ;
}
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_pbm )
2021-02-01 09:03:04 +00:00
{
2023-06-18 22:58:15 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " pnm/buggie-raw.pbm " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : PBMImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : PBMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-20 08:13:14 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_pgm )
2021-02-01 09:03:04 +00:00
{
2023-06-18 22:58:15 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " pnm/buggie-raw.pgm " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : PGMImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : PGMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-20 08:13:14 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_png )
2021-02-01 09:03:04 +00:00
{
2023-10-21 21:26:44 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " png/buggie.png " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : PNGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : PNGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2021-02-01 09:03:04 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2023-10-21 22:22:32 +00:00
TEST_CASE ( test_png_malformed_frame )
{
Array test_inputs = {
TEST_INPUT ( " png/oss-fuzz-testcase-62371.png " sv ) ,
TEST_INPUT ( " png/oss-fuzz-testcase-63052.png " sv )
} ;
for ( auto test_input : test_inputs ) {
auto file = MUST ( Core : : MappedFile : : map ( test_input ) ) ;
auto plugin_decoder = TRY_OR_FAIL ( Gfx : : PNGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
auto frame_or_error = plugin_decoder - > frame ( 0 ) ;
EXPECT ( frame_or_error . is_error ( ) ) ;
}
}
2021-04-29 01:38:16 +00:00
TEST_CASE ( test_ppm )
2021-02-01 09:03:04 +00:00
{
2023-06-18 22:58:15 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " pnm/buggie-raw.ppm " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : PPMImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-01-29 22:21:24 +00:00
auto plugin_decoder = MUST ( Gfx : : PPMImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-20 08:13:14 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2021-02-01 09:03:04 +00:00
}
2022-12-31 11:34:05 +00:00
TEST_CASE ( test_targa_bottom_left )
{
2023-06-18 22:56:41 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " tga/buggie-bottom-left-uncompressed.tga " sv ) ) ) ;
2023-01-29 22:21:24 +00:00
EXPECT ( MUST ( Gfx : : TGAImageDecoderPlugin : : validate_before_create ( file - > bytes ( ) ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TGAImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2022-12-31 11:34:05 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2022-12-31 11:34:05 +00:00
}
TEST_CASE ( test_targa_top_left )
{
2023-06-18 22:56:41 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " tga/buggie-top-left-uncompressed.tga " sv ) ) ) ;
2023-01-29 22:21:24 +00:00
EXPECT ( MUST ( Gfx : : TGAImageDecoderPlugin : : validate_before_create ( file - > bytes ( ) ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TGAImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-20 08:13:14 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2022-12-31 11:34:05 +00:00
}
2023-01-07 15:02:00 +00:00
TEST_CASE ( test_targa_bottom_left_compressed )
{
2023-06-18 22:56:41 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " tga/buggie-bottom-left-compressed.tga " sv ) ) ) ;
2023-01-29 22:21:24 +00:00
EXPECT ( MUST ( Gfx : : TGAImageDecoderPlugin : : validate_before_create ( file - > bytes ( ) ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TGAImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-07 15:02:00 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2023-01-07 15:02:00 +00:00
}
TEST_CASE ( test_targa_top_left_compressed )
{
2023-06-18 22:56:41 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " tga/buggie-top-left-compressed.tga " sv ) ) ) ;
2023-01-29 22:21:24 +00:00
EXPECT ( MUST ( Gfx : : TGAImageDecoderPlugin : : validate_before_create ( file - > bytes ( ) ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TGAImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-01-20 08:13:14 +00:00
2023-06-20 00:46:43 +00:00
expect_single_frame ( * plugin_decoder ) ;
2023-01-07 15:02:00 +00:00
}
2023-02-26 00:27:09 +00:00
TEST_CASE ( test_webp_simple_lossy )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/simple-vp8.webp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-26 00:27:09 +00:00
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 240 , 240 } ) ;
2023-05-29 17:01:45 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
EXPECT_EQ ( frame . image - > get_pixel ( 120 , 232 ) , Gfx : : Color ( 0xf2 , 0xef , 0xf0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 198 , 202 ) , Gfx : : Color ( 0x7b , 0xaa , 0xd5 , 255 ) ) ;
2023-02-26 00:27:09 +00:00
}
TEST_CASE ( test_webp_simple_lossless )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/simple-vp8l.webp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-26 00:27:09 +00:00
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-04-06 17:33:43 +00:00
// Ironically, simple-vp8l.webp is a much more complex file than extended-lossless.webp tested below.
// extended-lossless.webp tests the decoding basics.
// This here tests the predictor, color, and subtract green transforms,
// as well as meta prefix images, one-element canonical code handling,
// and handling of canonical codes with more than 288 elements.
// This image uses all 13 predictor modes of the predictor transform.
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 386 , 395 } ) ;
WebP/Lossless: Set alpha to 0xff if is_alpha_used is false in header
simple-vp8l-alpha-used-false.webp is a copy of simple-vp8l.webp,
with the byte at offset 0x18 changed from 0x10 to 0x00 -- that
is, the bit in the VP8L header that stores `is_alpha_used` is cleared.
We would already allocated a BGRx8888 instead of a BGRA8888 bitmap,
but keep actual alpha data in the `x` channel.
That lead to at least `image` still writing a PNG with an alpha channel.
So explicitly set the alpha channel to 0xff when is_alpha_used is false,
to make sure all consumers of decoded lossless webp data have behavior
consistent with other webp readers.
In practice, webp encoders usually don't write files that have
`is_alpha_used` set to false and then write actual alpha data to their
output. So this is rarely observable. However, for example for
lossy+ALPH webp files, the lossless webp used to store the ALPH channel
has `is_alpha_used` set to false and all channels but green are 0
(since the lossless green channel stores the alpha channel of a
lossy+ALPH webp). So if we dump such a bitmap to a standalone webp
file (e.g. with the temporary debugging code in fc3249a1ca8f42e72),
then without this commit here, `image` would convert that webp to
a fully transparent webp, while other webp software would correctly
display the green image with opaque alpha.
2023-06-18 14:47:05 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 0 , 0 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) ) ;
2023-04-06 17:33:43 +00:00
// This pixel tests all predictor modes except 5, 7, 8, 9, and 13.
EXPECT_EQ ( frame . image - > get_pixel ( 289 , 332 ) , Gfx : : Color ( 0xf2 , 0xee , 0xd3 , 255 ) ) ;
2023-02-26 00:27:09 +00:00
}
WebP/Lossless: Set alpha to 0xff if is_alpha_used is false in header
simple-vp8l-alpha-used-false.webp is a copy of simple-vp8l.webp,
with the byte at offset 0x18 changed from 0x10 to 0x00 -- that
is, the bit in the VP8L header that stores `is_alpha_used` is cleared.
We would already allocated a BGRx8888 instead of a BGRA8888 bitmap,
but keep actual alpha data in the `x` channel.
That lead to at least `image` still writing a PNG with an alpha channel.
So explicitly set the alpha channel to 0xff when is_alpha_used is false,
to make sure all consumers of decoded lossless webp data have behavior
consistent with other webp readers.
In practice, webp encoders usually don't write files that have
`is_alpha_used` set to false and then write actual alpha data to their
output. So this is rarely observable. However, for example for
lossy+ALPH webp files, the lossless webp used to store the ALPH channel
has `is_alpha_used` set to false and all channels but green are 0
(since the lossless green channel stores the alpha channel of a
lossy+ALPH webp). So if we dump such a bitmap to a standalone webp
file (e.g. with the temporary debugging code in fc3249a1ca8f42e72),
then without this commit here, `image` would convert that webp to
a fully transparent webp, while other webp software would correctly
display the green image with opaque alpha.
2023-06-18 14:47:05 +00:00
TEST_CASE ( test_webp_simple_lossless_alpha_used_false )
{
// This file is identical to simple-vp8l.webp, but the `is_alpha_used` used bit is false.
// The file still contains alpha data. This tests that the decoder replaces the stored alpha data with 0xff if `is_alpha_used` is false.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/simple-vp8l-alpha-used-false.webp " sv ) ) ) ;
WebP/Lossless: Set alpha to 0xff if is_alpha_used is false in header
simple-vp8l-alpha-used-false.webp is a copy of simple-vp8l.webp,
with the byte at offset 0x18 changed from 0x10 to 0x00 -- that
is, the bit in the VP8L header that stores `is_alpha_used` is cleared.
We would already allocated a BGRx8888 instead of a BGRA8888 bitmap,
but keep actual alpha data in the `x` channel.
That lead to at least `image` still writing a PNG with an alpha channel.
So explicitly set the alpha channel to 0xff when is_alpha_used is false,
to make sure all consumers of decoded lossless webp data have behavior
consistent with other webp readers.
In practice, webp encoders usually don't write files that have
`is_alpha_used` set to false and then write actual alpha data to their
output. So this is rarely observable. However, for example for
lossy+ALPH webp files, the lossless webp used to store the ALPH channel
has `is_alpha_used` set to false and all channels but green are 0
(since the lossless green channel stores the alpha channel of a
lossy+ALPH webp). So if we dump such a bitmap to a standalone webp
file (e.g. with the temporary debugging code in fc3249a1ca8f42e72),
then without this commit here, `image` would convert that webp to
a fully transparent webp, while other webp software would correctly
display the green image with opaque alpha.
2023-06-18 14:47:05 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 386 , 395 } ) ;
WebP/Lossless: Set alpha to 0xff if is_alpha_used is false in header
simple-vp8l-alpha-used-false.webp is a copy of simple-vp8l.webp,
with the byte at offset 0x18 changed from 0x10 to 0x00 -- that
is, the bit in the VP8L header that stores `is_alpha_used` is cleared.
We would already allocated a BGRx8888 instead of a BGRA8888 bitmap,
but keep actual alpha data in the `x` channel.
That lead to at least `image` still writing a PNG with an alpha channel.
So explicitly set the alpha channel to 0xff when is_alpha_used is false,
to make sure all consumers of decoded lossless webp data have behavior
consistent with other webp readers.
In practice, webp encoders usually don't write files that have
`is_alpha_used` set to false and then write actual alpha data to their
output. So this is rarely observable. However, for example for
lossy+ALPH webp files, the lossless webp used to store the ALPH channel
has `is_alpha_used` set to false and all channels but green are 0
(since the lossless green channel stores the alpha channel of a
lossy+ALPH webp). So if we dump such a bitmap to a standalone webp
file (e.g. with the temporary debugging code in fc3249a1ca8f42e72),
then without this commit here, `image` would convert that webp to
a fully transparent webp, while other webp software would correctly
display the green image with opaque alpha.
2023-06-18 14:47:05 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 0 , 0 ) , Gfx : : Color ( 0 , 0 , 0 , 0xff ) ) ;
}
2023-02-26 00:27:09 +00:00
TEST_CASE ( test_webp_extended_lossy )
{
2023-06-08 21:28:59 +00:00
// This extended lossy image has an ALPH chunk for (losslessly compressed) alpha data.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/extended-lossy.webp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-26 00:27:09 +00:00
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 417 , 223 } ) ;
2023-05-29 17:01:45 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
EXPECT_EQ ( frame . image - > get_pixel ( 89 , 72 ) , Gfx : : Color ( 255 , 1 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 174 , 69 ) , Gfx : : Color ( 0 , 255 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 245 , 84 ) , Gfx : : Color ( 0 , 0 , 255 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 352 , 125 ) , Gfx : : Color ( 0 , 0 , 0 , 128 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 355 , 106 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) ) ;
// Check same basic pixels as in test_webp_extended_lossless too.
// (The top-left pixel in the lossy version is fully transparent white, compared to fully transparent black in the lossless version).
EXPECT_EQ ( frame . image - > get_pixel ( 0 , 0 ) , Gfx : : Color ( 255 , 255 , 255 , 0 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 43 , 75 ) , Gfx : : Color ( 255 , 0 , 2 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 141 , 75 ) , Gfx : : Color ( 0 , 255 , 3 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 235 , 75 ) , Gfx : : Color ( 0 , 0 , 255 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 341 , 75 ) , Gfx : : Color ( 0 , 0 , 0 , 128 ) ) ;
2023-02-26 00:27:09 +00:00
}
2023-06-08 21:28:59 +00:00
TEST_CASE ( test_webp_extended_lossy_alpha_horizontal_filter )
{
// Also lossy rgb + lossless alpha, but with a horizontal alpha filtering method.
// The image should look like smolkling.webp, but with a horizontal alpha gradient.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/smolkling-horizontal-alpha.webp " sv ) ) ) ;
2023-06-08 21:28:59 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 264 , 264 } ) ;
2023-06-08 21:28:59 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
// The important component in this test is alpha, and that shouldn't change even by 1 as it's losslessly compressed and doesn't use YUV.
EXPECT_EQ ( frame . image - > get_pixel ( 131 , 131 ) , Gfx : : Color ( 0x8f , 0x51 , 0x2f , 0x4b ) ) ;
}
2023-06-08 22:28:11 +00:00
TEST_CASE ( test_webp_extended_lossy_alpha_vertical_filter )
{
// Also lossy rgb + lossless alpha, but with a vertical alpha filtering method.
// The image should look like smolkling.webp, but with a vertical alpha gradient, and with a fully transparent first column.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/smolkling-vertical-alpha.webp " sv ) ) ) ;
2023-06-08 22:28:11 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 264 , 264 } ) ;
2023-06-08 22:28:11 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
// The important component in this test is alpha, and that shouldn't change even by 1 as it's losslessly compressed and doesn't use YUV.
EXPECT_EQ ( frame . image - > get_pixel ( 131 , 131 ) , Gfx : : Color ( 0x94 , 0x50 , 0x32 , 0x4c ) ) ;
}
2023-06-08 21:55:39 +00:00
TEST_CASE ( test_webp_extended_lossy_alpha_gradient_filter )
{
// Also lossy rgb + lossless alpha, but with a gradient alpha filtering method.
// The image should look like smolkling.webp, but with a few transparent pixels in the shape of a C on it. Most of the image should not be transparent.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/smolkling-gradient-alpha.webp " sv ) ) ) ;
2023-06-08 21:55:39 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 264 , 264 } ) ;
2023-06-08 21:55:39 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
// The important component in this test is alpha, and that shouldn't change even by 1 as it's losslessly compressed and doesn't use YUV.
// In particular, the center of the image should be fully opaque, not fully transparent.
EXPECT_EQ ( frame . image - > get_pixel ( 131 , 131 ) , Gfx : : Color ( 0x8c , 0x47 , 0x2e , 255 ) ) ;
}
2023-05-29 23:19:29 +00:00
TEST_CASE ( test_webp_extended_lossy_uncompressed_alpha )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/extended-lossy-uncompressed-alpha.webp " sv ) ) ) ;
2023-05-29 23:19:29 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 417 , 223 } ) ;
2023-05-29 23:19:29 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
EXPECT_EQ ( frame . image - > get_pixel ( 89 , 72 ) , Gfx : : Color ( 255 , 0 , 4 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 174 , 69 ) , Gfx : : Color ( 4 , 255 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 245 , 84 ) , Gfx : : Color ( 0 , 0 , 255 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 352 , 125 ) , Gfx : : Color ( 0 , 0 , 0 , 128 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 355 , 106 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) ) ;
}
2023-06-01 13:09:24 +00:00
TEST_CASE ( test_webp_extended_lossy_negative_quantization_offset )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/smolkling.webp " sv ) ) ) ;
2023-06-01 13:09:24 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 264 , 264 } ) ;
2023-06-01 13:09:24 +00:00
// While VP8 YUV contents are defined bit-exact, the YUV->RGB conversion isn't.
// So pixels changing by 1 or so below is fine if you change code.
EXPECT_EQ ( frame . image - > get_pixel ( 16 , 16 ) , Gfx : : Color ( 0x3c , 0x24 , 0x1a , 255 ) ) ;
}
2023-05-30 14:21:58 +00:00
TEST_CASE ( test_webp_lossy_4 )
2023-05-29 19:23:33 +00:00
{
// This is https://commons.wikimedia.org/wiki/File:Fr%C3%BChling_bl%C3%BChender_Kirschenbaum.jpg,
// under the Creative Commons Attribution-Share Alike 3.0 Unported license. The image was re-encoded
// as webp at https://developers.google.com/speed/webp/gallery1 and the webp version is from there.
// No other changes have been made.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/4.webp " sv ) ) ) ;
2023-05-29 19:23:33 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 1024 , 772 } ) ;
2023-05-29 19:23:33 +00:00
// This image tests macroblocks that have `skip_coefficients` set to true, and it test a boolean entropy decoder edge case.
2023-05-30 14:21:58 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 780 , 570 ) , Gfx : : Color ( 0x72 , 0xc8 , 0xf6 , 255 ) ) ;
2023-05-29 19:23:33 +00:00
}
WebP/Lossy: Add support for images with more than one partition
Each secondary partition has an independent BooleanDecoder.
Their bitstreams interleave per macroblock row, that is the first
macroblock row is read from the first decoder, the second from the
second, ..., until it wraps around again.
All partitions share a single prediction state though: The second
macroblock row (which reads coefficients off the second decoder) is
predicted using the result of decoding the frist macroblock row (which
reads coefficients off the first decoder).
So if I understand things right, in theory the coefficient reading could
be parallelized, but prediction can't be. (IDCT can also be
parallelized, but that's true with just a single partition too.)
I created the test image by running
examples/cwebp -low_memory -partitions 3 -o foo.webp \
~/src/serenity/Tests/LibGfx/test-inputs/4.webp
using a cwebp hacked up as described in #19149. Since creating
multi-partition lossy webps requires hacking up `cwebp`, they're likely
very rare in practice. (But maybe other programs using the libwebp API
create them.)
Fixes #19149.
With this, webp lossy support is complete (*) :^)
And with that, webp support is complete: Lossless, lossy, lossy with
alpha, animated lossless, animated lossy, animated lossy with alpha all
work.
(*: Loop filtering isn't implemented yet, which has a minor visual
effect on the output. But it's only visible when carefully comparing
a webp decoded without loop filtering to the same decoded with it.
But it's technically a part of the spec that's still missing.
The upsampling of UV in the YUV->RGB code is also low-quality. This
produces somewhat visible banding in practice in some images (e.g.
in the fire breather's face in 5.webp), so we should probably improve
that at some point. Our JPG decoder has the same issue.)
2023-05-30 14:10:14 +00:00
TEST_CASE ( test_webp_lossy_4_with_partitions )
{
// Same input file as in the previous test, but re-encoded to use 8 secondary partitions.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/4-with-8-partitions.webp " sv ) ) ) ;
WebP/Lossy: Add support for images with more than one partition
Each secondary partition has an independent BooleanDecoder.
Their bitstreams interleave per macroblock row, that is the first
macroblock row is read from the first decoder, the second from the
second, ..., until it wraps around again.
All partitions share a single prediction state though: The second
macroblock row (which reads coefficients off the second decoder) is
predicted using the result of decoding the frist macroblock row (which
reads coefficients off the first decoder).
So if I understand things right, in theory the coefficient reading could
be parallelized, but prediction can't be. (IDCT can also be
parallelized, but that's true with just a single partition too.)
I created the test image by running
examples/cwebp -low_memory -partitions 3 -o foo.webp \
~/src/serenity/Tests/LibGfx/test-inputs/4.webp
using a cwebp hacked up as described in #19149. Since creating
multi-partition lossy webps requires hacking up `cwebp`, they're likely
very rare in practice. (But maybe other programs using the libwebp API
create them.)
Fixes #19149.
With this, webp lossy support is complete (*) :^)
And with that, webp support is complete: Lossless, lossy, lossy with
alpha, animated lossless, animated lossy, animated lossy with alpha all
work.
(*: Loop filtering isn't implemented yet, which has a minor visual
effect on the output. But it's only visible when carefully comparing
a webp decoded without loop filtering to the same decoded with it.
But it's technically a part of the spec that's still missing.
The upsampling of UV in the YUV->RGB code is also low-quality. This
produces somewhat visible banding in practice in some images (e.g.
in the fire breather's face in 5.webp), so we should probably improve
that at some point. Our JPG decoder has the same issue.)
2023-05-30 14:10:14 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 1024 , 772 } ) ;
WebP/Lossy: Add support for images with more than one partition
Each secondary partition has an independent BooleanDecoder.
Their bitstreams interleave per macroblock row, that is the first
macroblock row is read from the first decoder, the second from the
second, ..., until it wraps around again.
All partitions share a single prediction state though: The second
macroblock row (which reads coefficients off the second decoder) is
predicted using the result of decoding the frist macroblock row (which
reads coefficients off the first decoder).
So if I understand things right, in theory the coefficient reading could
be parallelized, but prediction can't be. (IDCT can also be
parallelized, but that's true with just a single partition too.)
I created the test image by running
examples/cwebp -low_memory -partitions 3 -o foo.webp \
~/src/serenity/Tests/LibGfx/test-inputs/4.webp
using a cwebp hacked up as described in #19149. Since creating
multi-partition lossy webps requires hacking up `cwebp`, they're likely
very rare in practice. (But maybe other programs using the libwebp API
create them.)
Fixes #19149.
With this, webp lossy support is complete (*) :^)
And with that, webp support is complete: Lossless, lossy, lossy with
alpha, animated lossless, animated lossy, animated lossy with alpha all
work.
(*: Loop filtering isn't implemented yet, which has a minor visual
effect on the output. But it's only visible when carefully comparing
a webp decoded without loop filtering to the same decoded with it.
But it's technically a part of the spec that's still missing.
The upsampling of UV in the YUV->RGB code is also low-quality. This
produces somewhat visible banding in practice in some images (e.g.
in the fire breather's face in 5.webp), so we should probably improve
that at some point. Our JPG decoder has the same issue.)
2023-05-30 14:10:14 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 780 , 570 ) , Gfx : : Color ( 0x73 , 0xc9 , 0xf9 , 255 ) ) ;
}
2023-02-26 00:27:09 +00:00
TEST_CASE ( test_webp_extended_lossless )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/extended-lossless.webp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-26 00:27:09 +00:00
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 417 , 223 } ) ;
2023-04-03 18:54:05 +00:00
// Check some basic pixels.
EXPECT_EQ ( frame . image - > get_pixel ( 0 , 0 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 43 , 75 ) , Gfx : : Color ( 255 , 0 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 141 , 75 ) , Gfx : : Color ( 0 , 255 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 235 , 75 ) , Gfx : : Color ( 0 , 0 , 255 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 341 , 75 ) , Gfx : : Color ( 0 , 0 , 0 , 128 ) ) ;
// Check pixels using the color cache.
EXPECT_EQ ( frame . image - > get_pixel ( 94 , 73 ) , Gfx : : Color ( 255 , 0 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 176 , 115 ) , Gfx : : Color ( 0 , 255 , 0 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 290 , 89 ) , Gfx : : Color ( 0 , 0 , 255 , 255 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 359 , 73 ) , Gfx : : Color ( 0 , 0 , 0 , 128 ) ) ;
2023-02-26 00:27:09 +00:00
}
2023-04-07 19:21:57 +00:00
TEST_CASE ( test_webp_simple_lossless_color_index_transform )
{
// In addition to testing the index transform, this file also tests handling of explicity setting max_symbol.
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/Qpalette.webp " sv ) ) ) ;
2023-04-07 19:21:57 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 256 , 256 } ) ;
2023-04-07 19:21:57 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 100 , 100 ) , Gfx : : Color ( 0x73 , 0x37 , 0x23 , 0xff ) ) ;
}
2023-04-07 22:40:28 +00:00
TEST_CASE ( test_webp_simple_lossless_color_index_transform_pixel_bundling )
{
struct TestCase {
StringView file_name ;
Gfx : : Color line_color ;
Gfx : : Color background_color ;
} ;
// The number after the dash is the number of colors in each file's color index bitmap.
// catdog-alert-2 tests the 1-bit-per-pixel case,
// catdog-alert-3 tests the 2-bit-per-pixel case,
// catdog-alert-8 and catdog-alert-13 both test the 4-bits-per-pixel case.
2023-06-20 01:07:17 +00:00
// catdog-alert-13-alpha-used-false is like catdog-alert-13, but with is_alpha_used set to false in the header
// (which has the effect of ignoring the alpha information in the palette and instead always setting alpha to 0xff).
2023-04-07 22:40:28 +00:00
TestCase test_cases [ ] = {
2023-06-18 22:34:25 +00:00
{ " webp/catdog-alert-2.webp " sv , Gfx : : Color ( 0x35 , 0x12 , 0x0a , 0xff ) , Gfx : : Color ( 0xf3 , 0xe6 , 0xd8 , 0xff ) } ,
{ " webp/catdog-alert-3.webp " sv , Gfx : : Color ( 0x35 , 0x12 , 0x0a , 0xff ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) } ,
{ " webp/catdog-alert-8.webp " sv , Gfx : : Color ( 0 , 0 , 0 , 255 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) } ,
{ " webp/catdog-alert-13.webp " sv , Gfx : : Color ( 0 , 0 , 0 , 255 ) , Gfx : : Color ( 0 , 0 , 0 , 0 ) } ,
2023-06-20 01:07:17 +00:00
{ " webp/catdog-alert-13-alpha-used-false.webp " sv , Gfx : : Color ( 0 , 0 , 0 , 255 ) , Gfx : : Color ( 0 , 0 , 0 , 255 ) } ,
2023-04-07 22:40:28 +00:00
} ;
for ( auto test_case : test_cases ) {
auto file = MUST ( Core : : MappedFile : : map ( MUST ( String : : formatted ( " {}{} " , TEST_INPUT ( " " ) , test_case . file_name ) ) ) ) ;
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
auto frame = expect_single_frame_of_size ( * plugin_decoder , { 32 , 32 } ) ;
2023-04-07 22:40:28 +00:00
EXPECT_EQ ( frame . image - > get_pixel ( 4 , 0 ) , test_case . background_color ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 5 , 0 ) , test_case . line_color ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 9 , 5 ) , test_case . background_color ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 10 , 5 ) , test_case . line_color ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 11 , 5 ) , test_case . background_color ) ;
}
}
2023-04-08 21:50:54 +00:00
TEST_CASE ( test_webp_simple_lossless_color_index_transform_pixel_bundling_odd_width )
{
StringView file_names [ ] = {
2023-06-18 22:34:25 +00:00
" webp/width11-height11-colors2.webp " sv ,
" webp/width11-height11-colors3.webp " sv ,
" webp/width11-height11-colors15.webp " sv ,
2023-04-08 21:50:54 +00:00
} ;
for ( auto file_name : file_names ) {
auto file = MUST ( Core : : MappedFile : : map ( MUST ( String : : formatted ( " {}{} " , TEST_INPUT ( " " ) , file_name ) ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-06-20 00:46:43 +00:00
expect_single_frame_of_size ( * plugin_decoder , { 11 , 11 } ) ;
2023-04-08 21:50:54 +00:00
}
}
2023-02-26 00:27:09 +00:00
TEST_CASE ( test_webp_extended_lossless_animated )
{
2023-06-18 22:34:25 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " webp/extended-lossless-animated.webp " sv ) ) ) ;
2023-02-26 18:02:50 +00:00
EXPECT ( Gfx : : WebPImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
2023-02-26 00:27:09 +00:00
auto plugin_decoder = MUST ( Gfx : : WebPImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
2023-02-26 03:15:34 +00:00
EXPECT_EQ ( plugin_decoder - > frame_count ( ) , 8u ) ;
EXPECT ( plugin_decoder - > is_animated ( ) ) ;
2023-02-26 14:13:27 +00:00
EXPECT_EQ ( plugin_decoder - > loop_count ( ) , 42u ) ;
2023-02-26 00:27:09 +00:00
EXPECT_EQ ( plugin_decoder - > size ( ) , Gfx : : IntSize ( 990 , 1050 ) ) ;
2023-05-06 20:28:26 +00:00
for ( size_t frame_index = 0 ; frame_index < plugin_decoder - > frame_count ( ) ; + + frame_index ) {
auto frame = MUST ( plugin_decoder - > frame ( frame_index ) ) ;
EXPECT_EQ ( frame . image - > size ( ) , Gfx : : IntSize ( 990 , 1050 ) ) ;
// This pixel happens to be the same color in all frames.
EXPECT_EQ ( frame . image - > get_pixel ( 500 , 700 ) , Gfx : : Color : : Yellow ) ;
// This one isn't the same in all frames.
EXPECT_EQ ( frame . image - > get_pixel ( 500 , 0 ) , ( frame_index = = 2 | | frame_index = = 6 ) ? Gfx : : Color : : Black : Gfx : : Color ( 255 , 255 , 255 , 0 ) ) ;
}
2023-02-26 00:27:09 +00:00
}
2023-07-02 22:09:27 +00:00
TEST_CASE ( test_tvg )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " tvg/yak.tvg " sv ) ) ) ;
EXPECT ( Gfx : : TinyVGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TinyVGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 1024 , 1024 } ) ;
}
2023-07-14 20:35:26 +00:00
TEST_CASE ( test_everything_tvg )
{
Array file_names {
TEST_INPUT ( " tvg/everything.tvg " sv ) ,
TEST_INPUT ( " tvg/everything-32.tvg " sv )
} ;
for ( auto file_name : file_names ) {
auto file = MUST ( Core : : MappedFile : : map ( file_name ) ) ;
EXPECT ( Gfx : : TinyVGImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : TinyVGImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 400 , 768 } ) ;
}
}
2023-07-22 03:04:29 +00:00
TEST_CASE ( test_jxl_modular_simple_tree_upsample2_10bits )
{
2023-07-22 17:20:48 +00:00
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jxl/modular_simple_tree_upsample2_10bits_rct.jxl " sv ) ) ) ;
2023-07-22 03:04:29 +00:00
EXPECT ( Gfx : : JPEGXLImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGXLImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 128 , 128 } ) ;
2023-07-22 17:20:48 +00:00
auto frame = MUST ( plugin_decoder - > frame ( 0 ) ) ;
EXPECT_EQ ( frame . image - > get_pixel ( 42 , 57 ) , Gfx : : Color : : from_string ( " #4c0072 " sv ) ) ;
2023-07-22 03:04:29 +00:00
}
2023-07-31 20:56:12 +00:00
TEST_CASE ( test_jxl_modular_property_8 )
{
auto file = MUST ( Core : : MappedFile : : map ( TEST_INPUT ( " jxl/modular_property_8.jxl " sv ) ) ) ;
EXPECT ( Gfx : : JPEGXLImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : JPEGXLImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame_of_size ( * plugin_decoder , { 32 , 32 } ) ;
auto frame = MUST ( plugin_decoder - > frame ( 0 ) ) ;
for ( u8 i = 0 ; i < 32 ; + + i ) {
for ( u8 j = 0 ; j < 32 ; + + j ) {
auto const color = frame . image - > get_pixel ( i , j ) ;
if ( ( i + j ) % 2 = = 0 )
EXPECT_EQ ( color , Gfx : : Color : : Black ) ;
else
EXPECT_EQ ( color , Gfx : : Color : : Yellow ) ;
}
}
}
2023-10-06 17:51:57 +00:00
TEST_CASE ( test_dds )
{
Array file_names = {
TEST_INPUT ( " dds/catdog-alert-29x29.dds " sv ) ,
TEST_INPUT ( " dds/catdog-alert-32x32.dds " sv )
} ;
for ( auto file_name : file_names ) {
auto file = MUST ( Core : : MappedFile : : map ( file_name ) ) ;
EXPECT ( Gfx : : DDSImageDecoderPlugin : : sniff ( file - > bytes ( ) ) ) ;
auto plugin_decoder = MUST ( Gfx : : DDSImageDecoderPlugin : : create ( file - > bytes ( ) ) ) ;
expect_single_frame ( * plugin_decoder ) ;
}
}