diff options
author | Andreas Kling <kling@serenityos.org> | 2020-11-13 11:58:00 +0100 |
---|---|---|
committer | Andreas Kling <kling@serenityos.org> | 2020-11-13 12:01:32 +0100 |
commit | 69518bd178ebfaabbafe53080b2fed9797c28dc7 (patch) | |
tree | cfb029c96fd5aec56a658362452953bb0e9c0990 | |
parent | f234b8c129c95857b48a9872506114878a0eaf7d (diff) | |
download | serenity-69518bd178ebfaabbafe53080b2fed9797c28dc7.zip |
LibGfx: Fail PPM decode if there's not enough pixel data in the input
Fixes #3820.
-rw-r--r-- | Libraries/LibGfx/PPMLoader.cpp | 10 |
1 files changed, 10 insertions, 0 deletions
diff --git a/Libraries/LibGfx/PPMLoader.cpp b/Libraries/LibGfx/PPMLoader.cpp index bb345e18a5..d891821877 100644 --- a/Libraries/LibGfx/PPMLoader.cpp +++ b/Libraries/LibGfx/PPMLoader.cpp @@ -28,6 +28,7 @@ #include <AK/Endian.h> #include <AK/LexicalPath.h> #include <AK/MappedFile.h> +#include <AK/ScopeGuard.h> #include <AK/StringBuilder.h> #include <string.h> @@ -266,6 +267,7 @@ static bool read_max_val(PPMLoadingContext& context, Streamer& streamer) static bool read_image_data(PPMLoadingContext& context, Streamer& streamer) { Vector<Gfx::Color> color_data; + color_data.ensure_capacity(context.width * context.height); if (context.type == PPMLoadingContext::P3_ASCII) { u16 red; @@ -303,6 +305,9 @@ static bool read_image_data(PPMLoadingContext& context, Streamer& streamer) } } + if (context.width * context.height != color_data.size()) + return false; + context.bitmap = Bitmap::create_purgeable(BitmapFormat::RGB32, { context.width, context.height }); size_t index = 0; @@ -322,6 +327,10 @@ static bool decode_ppm(PPMLoadingContext& context) if (context.state >= PPMLoadingContext::State::Decoded) return true; + auto error_guard = ArmedScopeGuard([&] { + context.state = PPMLoadingContext::State::Error; + }); + Streamer streamer(context.data, context.data_size); if (!read_magic_number(context, streamer)) @@ -351,6 +360,7 @@ static bool decode_ppm(PPMLoadingContext& context) if (!read_image_data(context, streamer)) return false; + error_guard.disarm(); context.state = PPMLoadingContext::State::Decoded; return true; } |