mirror of
https://github.com/webmproject/libwebp.git
synced 2026-04-09 14:22:31 +02:00
Add a fuzzer for ReadAnimatedImage
Bug: 496629074 Change-Id: Ie984f0eab67e8e6eda44abeedf9c13aa213dd340
This commit is contained in:
@@ -752,12 +752,24 @@ if(WEBP_BUILD_WEBP_JS)
|
|||||||
endif()
|
endif()
|
||||||
|
|
||||||
if(WEBP_BUILD_ANIM_UTILS)
|
if(WEBP_BUILD_ANIM_UTILS)
|
||||||
|
# anim_util
|
||||||
|
parse_makefile_am(${CMAKE_CURRENT_SOURCE_DIR}/examples "ANIM_UTIL_SRCS"
|
||||||
|
"anim_util_[^ ]*")
|
||||||
|
add_library(anim_util STATIC ${ANIM_UTIL_SRCS})
|
||||||
|
target_include_directories(
|
||||||
|
anim_util
|
||||||
|
PUBLIC
|
||||||
|
$<BUILD_INTERFACE:${CMAKE_CURRENT_SOURCE_DIR};${CMAKE_CURRENT_BINARY_DIR}/src>
|
||||||
|
$<INSTALL_INTERFACE:${CMAKE_INSTALL_INCLUDEDIR}>)
|
||||||
|
target_link_libraries(anim_util PUBLIC webp GIF::GIF)
|
||||||
|
|
||||||
# anim_diff
|
# anim_diff
|
||||||
parse_makefile_am(${CMAKE_CURRENT_SOURCE_DIR}/examples "ANIM_DIFF_SRCS"
|
parse_makefile_am(${CMAKE_CURRENT_SOURCE_DIR}/examples "ANIM_DIFF_SRCS"
|
||||||
"anim_diff")
|
"anim_diff")
|
||||||
add_executable(anim_diff ${ANIM_DIFF_SRCS})
|
add_executable(anim_diff ${ANIM_DIFF_SRCS})
|
||||||
target_link_libraries(
|
target_link_libraries(
|
||||||
anim_diff
|
anim_diff
|
||||||
|
anim_util
|
||||||
exampleutil
|
exampleutil
|
||||||
imagedec
|
imagedec
|
||||||
imageenc
|
imageenc
|
||||||
@@ -773,6 +785,7 @@ if(WEBP_BUILD_ANIM_UTILS)
|
|||||||
add_executable(anim_dump ${ANIM_DUMP_SRCS})
|
add_executable(anim_dump ${ANIM_DUMP_SRCS})
|
||||||
target_link_libraries(
|
target_link_libraries(
|
||||||
anim_dump
|
anim_dump
|
||||||
|
anim_util
|
||||||
exampleutil
|
exampleutil
|
||||||
imagedec
|
imagedec
|
||||||
imageenc
|
imageenc
|
||||||
|
|||||||
@@ -24,23 +24,32 @@ if BUILD_WEBPINFO
|
|||||||
endif
|
endif
|
||||||
|
|
||||||
noinst_LTLIBRARIES = libexample_util.la
|
noinst_LTLIBRARIES = libexample_util.la
|
||||||
|
noinst_LTLIBRARIES += libanim_util.la
|
||||||
|
|
||||||
libexample_util_la_SOURCES = example_util.c example_util.h
|
libexample_util_la_SOURCES = example_util.c example_util.h
|
||||||
libexample_util_la_LIBADD = ../src/libwebp.la
|
libexample_util_la_LIBADD = ../src/libwebp.la
|
||||||
|
|
||||||
anim_diff_SOURCES = anim_diff.c anim_util.c anim_util.h gifdec.c gifdec.h
|
libanim_util_la_SOURCES = anim_util.c anim_util.h gifdec.c gifdec.h
|
||||||
|
libanim_util_la_LIBADD =
|
||||||
|
libanim_util_la_LIBADD += ../src/libwebp.la
|
||||||
|
libanim_util_la_LIBADD += ../src/demux/libwebpdemux.la
|
||||||
|
libanim_util_la_LIBADD += $(GIF_LIBS)
|
||||||
|
|
||||||
|
anim_diff_SOURCES = anim_diff.c
|
||||||
anim_diff_CPPFLAGS = $(AM_CPPFLAGS) $(GIF_INCLUDES)
|
anim_diff_CPPFLAGS = $(AM_CPPFLAGS) $(GIF_INCLUDES)
|
||||||
anim_diff_LDADD =
|
anim_diff_LDADD =
|
||||||
anim_diff_LDADD += ../src/demux/libwebpdemux.la
|
anim_diff_LDADD += ../src/demux/libwebpdemux.la
|
||||||
|
anim_diff_LDADD += libanim_util.la
|
||||||
anim_diff_LDADD += libexample_util.la
|
anim_diff_LDADD += libexample_util.la
|
||||||
anim_diff_LDADD += ../imageio/libimageio_util.la
|
anim_diff_LDADD += ../imageio/libimageio_util.la
|
||||||
anim_diff_LDADD += $(GIF_LIBS) -lm
|
anim_diff_LDADD += $(GIF_LIBS) -lm
|
||||||
|
|
||||||
anim_dump_SOURCES = anim_dump.c anim_util.c anim_util.h gifdec.c gifdec.h
|
anim_dump_SOURCES = anim_dump.c
|
||||||
anim_dump_CPPFLAGS = $(AM_CPPFLAGS) $(PNG_INCLUDES)
|
anim_dump_CPPFLAGS = $(AM_CPPFLAGS) $(PNG_INCLUDES)
|
||||||
anim_dump_CPPFLAGS += $(GIF_INCLUDES)
|
anim_dump_CPPFLAGS += $(GIF_INCLUDES)
|
||||||
anim_dump_LDADD =
|
anim_dump_LDADD =
|
||||||
anim_dump_LDADD += ../src/demux/libwebpdemux.la
|
anim_dump_LDADD += ../src/demux/libwebpdemux.la
|
||||||
|
anim_dump_LDADD += libanim_util.la
|
||||||
anim_dump_LDADD += libexample_util.la
|
anim_dump_LDADD += libexample_util.la
|
||||||
anim_dump_LDADD += ../imageio/libimageio_util.la
|
anim_dump_LDADD += ../imageio/libimageio_util.la
|
||||||
anim_dump_LDADD += ../imageio/libimageenc.la
|
anim_dump_LDADD += ../imageio/libimageenc.la
|
||||||
|
|||||||
@@ -23,7 +23,6 @@
|
|||||||
#include "../imageio/imageio_util.h"
|
#include "../imageio/imageio_util.h"
|
||||||
#include "./gifdec.h"
|
#include "./gifdec.h"
|
||||||
#include "./unicode.h"
|
#include "./unicode.h"
|
||||||
#include "./unicode_gif.h"
|
|
||||||
#include "webp/decode.h"
|
#include "webp/decode.h"
|
||||||
#include "webp/demux.h"
|
#include "webp/demux.h"
|
||||||
#include "webp/format_constants.h"
|
#include "webp/format_constants.h"
|
||||||
@@ -294,6 +293,24 @@ End:
|
|||||||
|
|
||||||
#if defined(WEBP_HAVE_GIF)
|
#if defined(WEBP_HAVE_GIF)
|
||||||
|
|
||||||
|
typedef struct {
|
||||||
|
const uint8_t* data;
|
||||||
|
size_t size;
|
||||||
|
size_t offset;
|
||||||
|
} GifBufferContext;
|
||||||
|
|
||||||
|
static int MemoryReadGIF(GifFileType* gif, GifByteType* dest, int len) {
|
||||||
|
GifBufferContext* const ctx = (GifBufferContext*)gif->UserData;
|
||||||
|
if (ctx->offset + len > ctx->size) {
|
||||||
|
len = (int)(ctx->size - ctx->offset);
|
||||||
|
}
|
||||||
|
if (len > 0) {
|
||||||
|
memcpy(dest, ctx->data + ctx->offset, len);
|
||||||
|
ctx->offset += len;
|
||||||
|
}
|
||||||
|
return len;
|
||||||
|
}
|
||||||
|
|
||||||
// Returns true if this is a valid GIF bitstream.
|
// Returns true if this is a valid GIF bitstream.
|
||||||
static int IsGIF(const WebPData* const data) {
|
static int IsGIF(const WebPData* const data) {
|
||||||
return data->size > GIF_STAMP_LEN &&
|
return data->size > GIF_STAMP_LEN &&
|
||||||
@@ -504,18 +521,25 @@ static int ReadFrameGIF(const SavedImage* const gif_image,
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Read animated GIF bitstream from 'filename' into 'AnimatedImage' struct.
|
// Read animated GIF bitstream from 'gif_data' into 'AnimatedImage' struct.
|
||||||
static int ReadAnimatedGIF(const char filename[], AnimatedImage* const image,
|
static int ReadAnimatedGIF(const char filename[],
|
||||||
int dump_frames, const char dump_folder[]) {
|
const WebPData* const gif_data,
|
||||||
|
AnimatedImage* const image, int dump_frames,
|
||||||
|
const char dump_folder[]) {
|
||||||
uint32_t frame_count;
|
uint32_t frame_count;
|
||||||
uint32_t canvas_width, canvas_height;
|
uint32_t canvas_width, canvas_height;
|
||||||
uint32_t i;
|
uint32_t i;
|
||||||
int gif_error;
|
int gif_error;
|
||||||
GifFileType* gif;
|
GifFileType* gif;
|
||||||
|
GifBufferContext ctx;
|
||||||
|
|
||||||
gif = DGifOpenFileUnicode((const W_CHAR*)filename, NULL);
|
ctx.data = gif_data->bytes;
|
||||||
|
ctx.size = gif_data->size;
|
||||||
|
ctx.offset = 0;
|
||||||
|
gif = DGifOpen(&ctx, MemoryReadGIF, &gif_error);
|
||||||
if (gif == NULL) {
|
if (gif == NULL) {
|
||||||
WFPRINTF(stderr, "Could not read file: %s.\n", (const W_CHAR*)filename);
|
WFPRINTF(stderr, "Could not read GIF from memory: %s.\n",
|
||||||
|
(const W_CHAR*)filename);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -684,9 +708,12 @@ static int IsGIF(const WebPData* const data) {
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ReadAnimatedGIF(const char filename[], AnimatedImage* const image,
|
static int ReadAnimatedGIF(const char filename[],
|
||||||
int dump_frames, const char dump_folder[]) {
|
const WebPData* const gif_data,
|
||||||
|
AnimatedImage* const image, int dump_frames,
|
||||||
|
const char dump_folder[]) {
|
||||||
(void)filename;
|
(void)filename;
|
||||||
|
(void)gif_data;
|
||||||
(void)image;
|
(void)image;
|
||||||
(void)dump_frames;
|
(void)dump_frames;
|
||||||
(void)dump_folder;
|
(void)dump_folder;
|
||||||
@@ -706,18 +733,38 @@ int ReadAnimatedImage(const char filename[], AnimatedImage* const image,
|
|||||||
WebPData webp_data;
|
WebPData webp_data;
|
||||||
|
|
||||||
WebPDataInit(&webp_data);
|
WebPDataInit(&webp_data);
|
||||||
memset(image, 0, sizeof(*image));
|
|
||||||
|
|
||||||
if (!ImgIoUtilReadFile(filename, &webp_data.bytes, &webp_data.size)) {
|
if (!ImgIoUtilReadFile(filename, &webp_data.bytes, &webp_data.size)) {
|
||||||
WFPRINTF(stderr, "Error reading file: %s\n", (const W_CHAR*)filename);
|
WFPRINTF(stderr, "Error reading file: %s\n", (const W_CHAR*)filename);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
ok = ReadAnimatedImageFromMemory(filename, webp_data.bytes, webp_data.size,
|
||||||
|
image, dump_frames, dump_folder);
|
||||||
|
if (!ok) {
|
||||||
|
WFPRINTF(stderr, "Error parsing image: %s\n", (const W_CHAR*)filename);
|
||||||
|
}
|
||||||
|
|
||||||
|
WebPDataClear(&webp_data);
|
||||||
|
return ok;
|
||||||
|
}
|
||||||
|
|
||||||
|
int ReadAnimatedImageFromMemory(const char filename[],
|
||||||
|
const uint8_t* const data, size_t size,
|
||||||
|
AnimatedImage* const image, int dump_frames,
|
||||||
|
const char dump_folder[]) {
|
||||||
|
int ok = 0;
|
||||||
|
WebPData webp_data;
|
||||||
|
|
||||||
|
webp_data.bytes = data;
|
||||||
|
webp_data.size = size;
|
||||||
|
memset(image, 0, sizeof(*image));
|
||||||
|
|
||||||
if (IsWebP(&webp_data)) {
|
if (IsWebP(&webp_data)) {
|
||||||
ok =
|
ok =
|
||||||
ReadAnimatedWebP(filename, &webp_data, image, dump_frames, dump_folder);
|
ReadAnimatedWebP(filename, &webp_data, image, dump_frames, dump_folder);
|
||||||
} else if (IsGIF(&webp_data)) {
|
} else if (IsGIF(&webp_data)) {
|
||||||
ok = ReadAnimatedGIF(filename, image, dump_frames, dump_folder);
|
ok = ReadAnimatedGIF(filename, &webp_data, image, dump_frames, dump_folder);
|
||||||
} else {
|
} else {
|
||||||
WFPRINTF(stderr,
|
WFPRINTF(stderr,
|
||||||
"Unknown file type: %s. Supported file types are WebP and GIF\n",
|
"Unknown file type: %s. Supported file types are WebP and GIF\n",
|
||||||
@@ -725,7 +772,6 @@ int ReadAnimatedImage(const char filename[], AnimatedImage* const image,
|
|||||||
ok = 0;
|
ok = 0;
|
||||||
}
|
}
|
||||||
if (!ok) ClearAnimatedImage(image);
|
if (!ok) ClearAnimatedImage(image);
|
||||||
WebPDataClear(&webp_data);
|
|
||||||
return ok;
|
return ok;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -52,6 +52,13 @@ void ClearAnimatedImage(AnimatedImage* const image);
|
|||||||
int ReadAnimatedImage(const char filename[], AnimatedImage* const image,
|
int ReadAnimatedImage(const char filename[], AnimatedImage* const image,
|
||||||
int dump_frames, const char dump_folder[]);
|
int dump_frames, const char dump_folder[]);
|
||||||
|
|
||||||
|
// Same as 'ReadAnimatedImage', but from a memory buffer.
|
||||||
|
// filename is only used for log messages.
|
||||||
|
int ReadAnimatedImageFromMemory(const char filename[],
|
||||||
|
const uint8_t* const data, size_t size,
|
||||||
|
AnimatedImage* const image, int dump_frames,
|
||||||
|
const char dump_folder[]);
|
||||||
|
|
||||||
// Given two RGBA buffers, calculate max pixel difference and PSNR.
|
// Given two RGBA buffers, calculate max pixel difference and PSNR.
|
||||||
// If 'premultiply' is true, R/G/B values will be pre-multiplied by the
|
// If 'premultiply' is true, R/G/B values will be pre-multiplied by the
|
||||||
// transparency before comparison.
|
// transparency before comparison.
|
||||||
|
|||||||
@@ -59,6 +59,10 @@ add_webp_fuzztest(huffman_fuzzer webpdecode webpdspdecode webputilsdecode)
|
|||||||
add_webp_fuzztest(imageio_fuzzer imagedec)
|
add_webp_fuzztest(imageio_fuzzer imagedec)
|
||||||
add_webp_fuzztest(simple_api_fuzzer)
|
add_webp_fuzztest(simple_api_fuzzer)
|
||||||
|
|
||||||
|
if (WEBP_BUILD_ANIM_UTILS)
|
||||||
|
add_webp_fuzztest(anim_util_fuzzer anim_util imageioutil webpdemux)
|
||||||
|
endif()
|
||||||
|
|
||||||
if(WEBP_BUILD_LIBWEBPMUX)
|
if(WEBP_BUILD_LIBWEBPMUX)
|
||||||
add_webp_fuzztest(animation_api_fuzzer webpdemux)
|
add_webp_fuzztest(animation_api_fuzzer webpdemux)
|
||||||
add_webp_fuzztest(animdecoder_fuzzer imageioutil webpdemux)
|
add_webp_fuzztest(animdecoder_fuzzer imageioutil webpdemux)
|
||||||
|
|||||||
49
tests/fuzzer/anim_util_fuzzer.cc
Normal file
49
tests/fuzzer/anim_util_fuzzer.cc
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
// Copyright 2026 Google Inc.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
//
|
||||||
|
////////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
#include <cstdio>
|
||||||
|
#include <cstdlib>
|
||||||
|
#include <string_view>
|
||||||
|
|
||||||
|
#include "./fuzz_utils.h"
|
||||||
|
#include "./nalloc.h"
|
||||||
|
#include "examples/anim_util.h"
|
||||||
|
|
||||||
|
namespace {
|
||||||
|
|
||||||
|
void ReadAnimatedImageTest(std::string_view blob) {
|
||||||
|
const uint8_t* const data = reinterpret_cast<const uint8_t*>(blob.data());
|
||||||
|
const size_t size = blob.size();
|
||||||
|
if (fuzz_utils::IsImageTooBig(data, size)) return;
|
||||||
|
|
||||||
|
nalloc_init(nullptr);
|
||||||
|
nalloc_start(data, size);
|
||||||
|
|
||||||
|
AnimatedImage image;
|
||||||
|
if (ReadAnimatedImageFromMemory("random_file", data, size, &image,
|
||||||
|
/*dump_frames=*/0, /*dump_folder=*/nullptr)) {
|
||||||
|
ClearAnimatedImage(&image);
|
||||||
|
}
|
||||||
|
|
||||||
|
nalloc_end();
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
FUZZ_TEST(ReadAnimatedImage, ReadAnimatedImageTest)
|
||||||
|
.WithDomains(fuzztest::String().WithMaxSize(fuzz_utils::kMaxWebPFileSize +
|
||||||
|
1));
|
||||||
Reference in New Issue
Block a user