mirror of
				https://github.com/LadybirdBrowser/ladybird.git
				synced 2025-11-04 07:10:57 +00:00 
			
		
		
		
	An image with an incorrect header should fail at this step, so we have to handle that without crashing.
		
			
				
	
	
		
			103 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			103 lines
		
	
	
	
		
			4.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
/*
 | 
						|
 * Copyright (c) 2018-2021, Andreas Kling <kling@serenityos.org>
 | 
						|
 *
 | 
						|
 * SPDX-License-Identifier: BSD-2-Clause
 | 
						|
 */
 | 
						|
 | 
						|
#include <AK/LexicalPath.h>
 | 
						|
#include <LibGfx/ImageFormats/BMPLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/DDSLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/GIFLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/ICOLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/ImageDecoder.h>
 | 
						|
#include <LibGfx/ImageFormats/JPEGLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/PBMLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/PGMLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/PNGLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/PPMLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/QOILoader.h>
 | 
						|
#include <LibGfx/ImageFormats/TGALoader.h>
 | 
						|
#include <LibGfx/ImageFormats/TinyVGLoader.h>
 | 
						|
#include <LibGfx/ImageFormats/WebPLoader.h>
 | 
						|
 | 
						|
namespace Gfx {
 | 
						|
 | 
						|
static OwnPtr<ImageDecoderPlugin> probe_and_sniff_for_appropriate_plugin(ReadonlyBytes bytes)
 | 
						|
{
 | 
						|
    struct ImagePluginInitializer {
 | 
						|
        bool (*sniff)(ReadonlyBytes) = nullptr;
 | 
						|
        ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> (*create)(ReadonlyBytes) = nullptr;
 | 
						|
    };
 | 
						|
 | 
						|
    static constexpr ImagePluginInitializer s_initializers[] = {
 | 
						|
        { PNGImageDecoderPlugin::sniff, PNGImageDecoderPlugin::create },
 | 
						|
        { GIFImageDecoderPlugin::sniff, GIFImageDecoderPlugin::create },
 | 
						|
        { BMPImageDecoderPlugin::sniff, BMPImageDecoderPlugin::create },
 | 
						|
        { PBMImageDecoderPlugin::sniff, PBMImageDecoderPlugin::create },
 | 
						|
        { PGMImageDecoderPlugin::sniff, PGMImageDecoderPlugin::create },
 | 
						|
        { PPMImageDecoderPlugin::sniff, PPMImageDecoderPlugin::create },
 | 
						|
        { ICOImageDecoderPlugin::sniff, ICOImageDecoderPlugin::create },
 | 
						|
        { JPEGImageDecoderPlugin::sniff, JPEGImageDecoderPlugin::create },
 | 
						|
        { DDSImageDecoderPlugin::sniff, DDSImageDecoderPlugin::create },
 | 
						|
        { QOIImageDecoderPlugin::sniff, QOIImageDecoderPlugin::create },
 | 
						|
        { TinyVGImageDecoderPlugin::sniff, TinyVGImageDecoderPlugin::create },
 | 
						|
        { WebPImageDecoderPlugin::sniff, WebPImageDecoderPlugin::create },
 | 
						|
    };
 | 
						|
 | 
						|
    for (auto& plugin : s_initializers) {
 | 
						|
        auto sniff_result = plugin.sniff(bytes);
 | 
						|
        if (!sniff_result)
 | 
						|
            continue;
 | 
						|
        auto plugin_decoder = plugin.create(bytes);
 | 
						|
        if (!plugin_decoder.is_error()) {
 | 
						|
            if (!plugin_decoder.value()->initialize().is_error())
 | 
						|
                return plugin_decoder.release_value();
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return {};
 | 
						|
}
 | 
						|
 | 
						|
static OwnPtr<ImageDecoderPlugin> probe_and_sniff_for_appropriate_plugin_with_known_mime_type(StringView mime_type, ReadonlyBytes bytes)
 | 
						|
{
 | 
						|
    struct ImagePluginWithMIMETypeInitializer {
 | 
						|
        ErrorOr<bool> (*validate_before_create)(ReadonlyBytes) = nullptr;
 | 
						|
        ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> (*create)(ReadonlyBytes) = nullptr;
 | 
						|
        StringView mime_type;
 | 
						|
    };
 | 
						|
 | 
						|
    static constexpr ImagePluginWithMIMETypeInitializer s_initializers_with_mime_type[] = {
 | 
						|
        { TGAImageDecoderPlugin::validate_before_create, TGAImageDecoderPlugin::create, "image/x-targa"sv },
 | 
						|
    };
 | 
						|
 | 
						|
    for (auto& plugin : s_initializers_with_mime_type) {
 | 
						|
        if (plugin.mime_type != mime_type)
 | 
						|
            continue;
 | 
						|
        auto validation_result = plugin.validate_before_create(bytes).release_value_but_fixme_should_propagate_errors();
 | 
						|
        if (!validation_result)
 | 
						|
            continue;
 | 
						|
        auto plugin_decoder = plugin.create(bytes).release_value_but_fixme_should_propagate_errors();
 | 
						|
        if (!plugin_decoder->initialize().is_error())
 | 
						|
            return plugin_decoder;
 | 
						|
    }
 | 
						|
    return {};
 | 
						|
}
 | 
						|
 | 
						|
RefPtr<ImageDecoder> ImageDecoder::try_create_for_raw_bytes(ReadonlyBytes bytes, Optional<DeprecatedString> mime_type)
 | 
						|
{
 | 
						|
    if (OwnPtr<ImageDecoderPlugin> plugin = probe_and_sniff_for_appropriate_plugin(bytes); plugin)
 | 
						|
        return adopt_ref_if_nonnull(new (nothrow) ImageDecoder(plugin.release_nonnull()));
 | 
						|
 | 
						|
    if (mime_type.has_value()) {
 | 
						|
        if (OwnPtr<ImageDecoderPlugin> plugin = probe_and_sniff_for_appropriate_plugin_with_known_mime_type(mime_type.value(), bytes); plugin)
 | 
						|
            return adopt_ref_if_nonnull(new (nothrow) ImageDecoder(plugin.release_nonnull()));
 | 
						|
    }
 | 
						|
 | 
						|
    return {};
 | 
						|
}
 | 
						|
 | 
						|
ImageDecoder::ImageDecoder(NonnullOwnPtr<ImageDecoderPlugin> plugin)
 | 
						|
    : m_plugin(move(plugin))
 | 
						|
{
 | 
						|
}
 | 
						|
 | 
						|
}
 |