mirror of
https://github.com/LadybirdBrowser/ladybird.git
synced 2025-10-24 00:49:46 +00:00
This has come up several times during code review, so let's just enforce it using a new clang-format 20 option.
274 lines
8.1 KiB
C++
274 lines
8.1 KiB
C++
/*
|
|
* Copyright (c) 2023-2024, Lucas Chollet <lucas.chollet@serenityos.org>
|
|
*
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
|
*/
|
|
|
|
#include <AK/Error.h>
|
|
#include <LibGfx/ImageFormats/JPEGXLLoader.h>
|
|
#include <jxl/decode.h>
|
|
|
|
namespace Gfx {
|
|
|
|
class JPEGXLLoadingContext {
|
|
AK_MAKE_NONCOPYABLE(JPEGXLLoadingContext);
|
|
AK_MAKE_NONMOVABLE(JPEGXLLoadingContext);
|
|
|
|
public:
|
|
JPEGXLLoadingContext(JxlDecoder* decoder)
|
|
: m_decoder(decoder)
|
|
{
|
|
}
|
|
|
|
~JPEGXLLoadingContext()
|
|
{
|
|
JxlDecoderDestroy(m_decoder);
|
|
}
|
|
|
|
ErrorOr<void> decode_image_header()
|
|
{
|
|
return run_state_machine_until(State::HeaderDecoded);
|
|
}
|
|
|
|
ErrorOr<void> decode_image()
|
|
{
|
|
return run_state_machine_until(State::FrameDecoded);
|
|
}
|
|
|
|
enum class State : u8 {
|
|
NotDecoded = 0,
|
|
Error,
|
|
HeaderDecoded,
|
|
FrameDecoded,
|
|
};
|
|
|
|
State state() const
|
|
{
|
|
return m_state;
|
|
}
|
|
|
|
IntSize size() const
|
|
{
|
|
return m_size;
|
|
}
|
|
|
|
Vector<ImageFrameDescriptor> const& frame_descriptors() const
|
|
{
|
|
return m_frame_descriptors;
|
|
}
|
|
|
|
bool is_animated() const
|
|
{
|
|
return m_animated;
|
|
}
|
|
|
|
u32 loop_count() const
|
|
{
|
|
return m_loop_count;
|
|
}
|
|
|
|
u32 frame_count() const
|
|
{
|
|
return m_frame_count;
|
|
}
|
|
|
|
private:
|
|
ErrorOr<void> run_state_machine_until(State requested_state)
|
|
{
|
|
Optional<u32> frame_duration;
|
|
for (;;) {
|
|
auto const status = JxlDecoderProcessInput(m_decoder);
|
|
|
|
if (status == JXL_DEC_ERROR)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Decoder is corrupted.");
|
|
if (status == JXL_DEC_NEED_MORE_INPUT)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Decoder need more input.");
|
|
|
|
if (status == JXL_DEC_BASIC_INFO) {
|
|
TRY(decode_image_header_impl());
|
|
if (requested_state <= State::HeaderDecoded)
|
|
return {};
|
|
}
|
|
|
|
if (status == JXL_DEC_FRAME) {
|
|
JxlFrameHeader header;
|
|
if (auto res = JxlDecoderGetFrameHeader(m_decoder, &header);
|
|
res != JXL_DEC_SUCCESS)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unable to retrieve frame header.");
|
|
|
|
frame_duration = header.duration;
|
|
continue;
|
|
}
|
|
|
|
if (status == JXL_DEC_NEED_IMAGE_OUT_BUFFER) {
|
|
if (!frame_duration.has_value())
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: No frame header was read.");
|
|
|
|
TRY(set_output_buffer(*frame_duration));
|
|
continue;
|
|
}
|
|
|
|
if (status == JXL_DEC_FULL_IMAGE) {
|
|
m_frame_count++;
|
|
continue;
|
|
}
|
|
|
|
if (status == JXL_DEC_SUCCESS) {
|
|
if (m_state != State::Error)
|
|
m_state = State::FrameDecoded;
|
|
return {};
|
|
}
|
|
|
|
warnln("JPEGXLImageDecoderPlugin: Unknown event.");
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unknown event.");
|
|
}
|
|
}
|
|
|
|
ErrorOr<void> decode_image_header_impl()
|
|
{
|
|
JxlBasicInfo info;
|
|
|
|
if (auto res = JxlDecoderGetBasicInfo(m_decoder, &info); res != JXL_DEC_SUCCESS)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unable to decode basic information.");
|
|
|
|
m_size = { info.xsize, info.ysize };
|
|
|
|
m_animated = static_cast<bool>(info.have_animation);
|
|
m_alpha_premultiplied = info.alpha_premultiplied ? Gfx::AlphaType::Premultiplied : Gfx::AlphaType::Unpremultiplied;
|
|
|
|
if (m_animated)
|
|
m_loop_count = info.animation.num_loops;
|
|
|
|
m_state = State::HeaderDecoded;
|
|
return {};
|
|
}
|
|
|
|
ErrorOr<void> set_output_buffer(u32 duration)
|
|
{
|
|
auto result = [this, duration]() -> ErrorOr<void> {
|
|
if (JxlDecoderProcessInput(m_decoder) != JXL_DEC_NEED_IMAGE_OUT_BUFFER)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Decoder is in an unexpected state.");
|
|
|
|
auto bitmap = TRY(Bitmap::create(Gfx::BitmapFormat::RGBA8888, m_alpha_premultiplied, m_size));
|
|
TRY(m_frame_descriptors.try_empend(bitmap, static_cast<int>(duration)));
|
|
|
|
JxlPixelFormat format = { 4, JXL_TYPE_UINT8, JXL_NATIVE_ENDIAN, 0 };
|
|
|
|
size_t needed_size = 0;
|
|
JxlDecoderImageOutBufferSize(m_decoder, &format, &needed_size);
|
|
|
|
if (needed_size != bitmap->size_in_bytes())
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Expected bitmap size is wrong.");
|
|
|
|
if (auto res = JxlDecoderSetImageOutBuffer(m_decoder, &format, bitmap->begin(), bitmap->size_in_bytes());
|
|
res != JXL_DEC_SUCCESS)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unable to decode frame.");
|
|
|
|
return {};
|
|
}();
|
|
|
|
if (result.is_error()) {
|
|
m_state = State::Error;
|
|
warnln("{}", result.error());
|
|
}
|
|
|
|
return result;
|
|
}
|
|
|
|
State m_state { State::NotDecoded };
|
|
|
|
JxlDecoder* m_decoder;
|
|
|
|
IntSize m_size;
|
|
Vector<ImageFrameDescriptor> m_frame_descriptors;
|
|
|
|
bool m_animated { false };
|
|
Gfx::AlphaType m_alpha_premultiplied { Gfx::AlphaType::Premultiplied };
|
|
u32 m_loop_count { 0 };
|
|
u32 m_frame_count { 0 };
|
|
};
|
|
|
|
JPEGXLImageDecoderPlugin::JPEGXLImageDecoderPlugin(OwnPtr<JPEGXLLoadingContext> context)
|
|
: m_context(move(context))
|
|
{
|
|
}
|
|
|
|
JPEGXLImageDecoderPlugin::~JPEGXLImageDecoderPlugin() = default;
|
|
|
|
IntSize JPEGXLImageDecoderPlugin::size()
|
|
{
|
|
return m_context->size();
|
|
}
|
|
|
|
bool JPEGXLImageDecoderPlugin::sniff(ReadonlyBytes data)
|
|
{
|
|
auto signature = JxlSignatureCheck(data.data(), data.size());
|
|
return signature == JXL_SIG_CODESTREAM || signature == JXL_SIG_CONTAINER;
|
|
}
|
|
|
|
ErrorOr<NonnullOwnPtr<ImageDecoderPlugin>> JPEGXLImageDecoderPlugin::create(ReadonlyBytes data)
|
|
{
|
|
auto* decoder = JxlDecoderCreate(nullptr);
|
|
if (!decoder)
|
|
return Error::from_errno(ENOMEM);
|
|
|
|
auto const events = JXL_DEC_BASIC_INFO | JXL_DEC_FRAME | JXL_DEC_FULL_IMAGE;
|
|
if (auto res = JxlDecoderSubscribeEvents(decoder, events); res == JXL_DEC_ERROR)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unable to subscribe to events.");
|
|
|
|
if (auto res = JxlDecoderSetInput(decoder, data.data(), data.size()); res == JXL_DEC_ERROR)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Unable to set decoder input.");
|
|
|
|
// Tell the decoder that it won't receive more data for the image.
|
|
JxlDecoderCloseInput(decoder);
|
|
|
|
auto context = TRY(adopt_nonnull_own_or_enomem(new (nothrow) JPEGXLLoadingContext(decoder)));
|
|
auto plugin = TRY(adopt_nonnull_own_or_enomem(new (nothrow) JPEGXLImageDecoderPlugin(move(context))));
|
|
|
|
TRY(plugin->m_context->decode_image_header());
|
|
return plugin;
|
|
}
|
|
|
|
bool JPEGXLImageDecoderPlugin::is_animated()
|
|
{
|
|
return m_context->is_animated();
|
|
}
|
|
|
|
size_t JPEGXLImageDecoderPlugin::loop_count()
|
|
{
|
|
return m_context->loop_count();
|
|
}
|
|
|
|
size_t JPEGXLImageDecoderPlugin::frame_count()
|
|
{
|
|
// FIXME: There doesn't seem to be a way to have that information
|
|
// before decoding all the frames.
|
|
if (m_context->frame_count() == 0)
|
|
(void)frame(0);
|
|
return m_context->frame_count();
|
|
}
|
|
|
|
size_t JPEGXLImageDecoderPlugin::first_animated_frame_index()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
ErrorOr<ImageFrameDescriptor> JPEGXLImageDecoderPlugin::frame(size_t index, Optional<IntSize>)
|
|
{
|
|
if (m_context->state() == JPEGXLLoadingContext::State::Error)
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Decoding failed.");
|
|
|
|
if (m_context->state() < JPEGXLLoadingContext::State::FrameDecoded)
|
|
TRY(m_context->decode_image());
|
|
|
|
if (index >= m_context->frame_descriptors().size())
|
|
return Error::from_string_literal("JPEGXLImageDecoderPlugin: Invalid frame index requested.");
|
|
return m_context->frame_descriptors()[index];
|
|
}
|
|
|
|
ErrorOr<Optional<ReadonlyBytes>> JPEGXLImageDecoderPlugin::icc_data()
|
|
{
|
|
return OptionalNone {};
|
|
}
|
|
|
|
}
|