2020-06-22 21:35:22 +02:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2020, Andreas Kling <kling@serenityos.org>
|
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-06-22 21:35:22 +02:00
|
|
|
*/
|
|
|
|
|
2021-04-21 20:08:03 +02:00
|
|
|
#include <AK/Debug.h>
|
2022-02-25 12:18:30 +02:00
|
|
|
#include <ImageDecoder/ConnectionFromClient.h>
|
2020-06-22 21:35:22 +02:00
|
|
|
#include <ImageDecoder/ImageDecoderClientEndpoint.h>
|
|
|
|
#include <LibGfx/Bitmap.h>
|
2023-03-21 14:58:06 -04:00
|
|
|
#include <LibGfx/ImageFormats/ImageDecoder.h>
|
2024-02-14 01:15:06 -05:00
|
|
|
#include <LibGfx/ImageFormats/TIFFMetadata.h>
|
2020-06-22 21:35:22 +02:00
|
|
|
|
|
|
|
namespace ImageDecoder {
|
|
|
|
|
2023-02-08 23:05:44 +01:00
|
|
|
ConnectionFromClient::ConnectionFromClient(NonnullOwnPtr<Core::LocalSocket> socket)
|
2022-02-25 12:18:30 +02:00
|
|
|
: IPC::ConnectionFromClient<ImageDecoderClientEndpoint, ImageDecoderServerEndpoint>(*this, move(socket), 1)
|
2020-06-22 21:35:22 +02:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2022-02-25 12:18:30 +02:00
|
|
|
void ConnectionFromClient::die()
|
2020-06-22 21:35:22 +02:00
|
|
|
{
|
2021-11-29 17:12:55 +01:00
|
|
|
Core::EventLoop::current().quit(0);
|
2020-06-22 21:35:22 +02:00
|
|
|
}
|
|
|
|
|
2023-12-23 13:56:23 -05:00
|
|
|
static void decode_image_to_bitmaps_and_durations_with_decoder(Gfx::ImageDecoder const& decoder, Optional<Gfx::IntSize> ideal_size, Vector<Gfx::ShareableBitmap>& bitmaps, Vector<u32>& durations)
|
2020-06-22 21:35:22 +02:00
|
|
|
{
|
2023-01-15 21:50:32 +02:00
|
|
|
for (size_t i = 0; i < decoder.frame_count(); ++i) {
|
2023-12-23 13:56:23 -05:00
|
|
|
auto frame_or_error = decoder.frame(i, ideal_size);
|
2023-01-15 21:50:32 +02:00
|
|
|
if (frame_or_error.is_error()) {
|
|
|
|
bitmaps.append(Gfx::ShareableBitmap {});
|
|
|
|
durations.append(0);
|
|
|
|
} else {
|
|
|
|
auto frame = frame_or_error.release_value();
|
|
|
|
bitmaps.append(frame.image->to_shareable_bitmap());
|
|
|
|
durations.append(frame.duration);
|
|
|
|
}
|
2020-06-22 21:35:22 +02:00
|
|
|
}
|
2023-01-15 21:50:32 +02:00
|
|
|
}
|
|
|
|
|
2024-02-14 01:15:06 -05:00
|
|
|
static void decode_image_to_details(Core::AnonymousBuffer const& encoded_buffer, Optional<Gfx::IntSize> ideal_size, Optional<ByteString> const& known_mime_type, bool& is_animated, u32& loop_count, Vector<Gfx::ShareableBitmap>& bitmaps, Vector<u32>& durations, Gfx::FloatPoint& scale)
|
2023-01-15 21:50:32 +02:00
|
|
|
{
|
|
|
|
VERIFY(bitmaps.size() == 0);
|
|
|
|
VERIFY(durations.size() == 0);
|
2020-06-22 21:35:22 +02:00
|
|
|
|
2024-03-04 18:07:43 -05:00
|
|
|
auto decoder_or_err = Gfx::ImageDecoder::try_create_for_raw_bytes(ReadonlyBytes { encoded_buffer.data<u8>(), encoded_buffer.size() }, known_mime_type);
|
|
|
|
if (decoder_or_err.is_error() || !decoder_or_err.value()) {
|
2021-07-27 01:12:53 +02:00
|
|
|
dbgln_if(IMAGE_DECODER_DEBUG, "Could not find suitable image decoder plugin for data");
|
2023-01-15 21:50:32 +02:00
|
|
|
return;
|
2021-07-27 01:12:53 +02:00
|
|
|
}
|
2024-03-04 18:07:43 -05:00
|
|
|
auto decoder = decoder_or_err.release_value();
|
2020-06-22 21:35:22 +02:00
|
|
|
|
2021-01-29 22:30:48 +01:00
|
|
|
if (!decoder->frame_count()) {
|
2021-04-21 20:08:03 +02:00
|
|
|
dbgln_if(IMAGE_DECODER_DEBUG, "Could not decode image from encoded data");
|
2023-01-15 21:50:32 +02:00
|
|
|
return;
|
2020-06-22 21:35:22 +02:00
|
|
|
}
|
2023-01-22 11:58:36 +01:00
|
|
|
is_animated = decoder->is_animated();
|
|
|
|
loop_count = decoder->loop_count();
|
2024-02-14 01:15:06 -05:00
|
|
|
|
|
|
|
if (auto maybe_metadata = decoder->metadata(); maybe_metadata.has_value() && is<Gfx::ExifMetadata>(*maybe_metadata)) {
|
|
|
|
auto const& exif = static_cast<Gfx::ExifMetadata const&>(maybe_metadata.value());
|
|
|
|
if (exif.x_resolution().has_value() && exif.y_resolution().has_value()) {
|
|
|
|
auto const x_resolution = exif.x_resolution()->as_double();
|
|
|
|
auto const y_resolution = exif.y_resolution()->as_double();
|
|
|
|
if (x_resolution < y_resolution)
|
|
|
|
scale.set_y(x_resolution / y_resolution);
|
|
|
|
else
|
|
|
|
scale.set_x(y_resolution / x_resolution);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-23 13:56:23 -05:00
|
|
|
decode_image_to_bitmaps_and_durations_with_decoder(*decoder, ideal_size, bitmaps, durations);
|
2023-01-15 21:50:32 +02:00
|
|
|
}
|
2020-06-22 21:35:22 +02:00
|
|
|
|
2024-04-19 14:23:16 -06:00
|
|
|
ConnectionFromClient::Job ConnectionFromClient::make_decode_image_job(i64 image_id, Core::AnonymousBuffer encoded_buffer, Optional<Gfx::IntSize> ideal_size, Optional<ByteString> mime_type)
|
|
|
|
{
|
|
|
|
return [strong_this = NonnullRefPtr(*this), image_id, encoded_buffer = move(encoded_buffer), ideal_size = move(ideal_size), mime_type = move(mime_type)]() {
|
|
|
|
bool is_animated = false;
|
|
|
|
u32 loop_count = 0;
|
|
|
|
Gfx::FloatPoint scale { 1, 1 };
|
|
|
|
Vector<Gfx::ShareableBitmap> bitmaps;
|
|
|
|
Vector<u32> durations;
|
|
|
|
|
|
|
|
decode_image_to_details(encoded_buffer, ideal_size, mime_type, is_animated, loop_count, bitmaps, durations, scale);
|
|
|
|
if (bitmaps.is_empty()) {
|
|
|
|
strong_this->async_did_fail_to_decode_image(image_id, "Could not decode image"_string);
|
|
|
|
} else {
|
|
|
|
strong_this->async_did_decode_image(image_id, is_animated, loop_count, bitmaps, durations, scale);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2023-12-23 13:56:23 -05:00
|
|
|
Messages::ImageDecoderServer::DecodeImageResponse ConnectionFromClient::decode_image(Core::AnonymousBuffer const& encoded_buffer, Optional<Gfx::IntSize> const& ideal_size, Optional<ByteString> const& mime_type)
|
2023-01-15 21:50:32 +02:00
|
|
|
{
|
2024-04-19 14:23:16 -06:00
|
|
|
auto image_id = m_next_image_id++;
|
|
|
|
|
2023-01-15 21:50:32 +02:00
|
|
|
if (!encoded_buffer.is_valid()) {
|
|
|
|
dbgln_if(IMAGE_DECODER_DEBUG, "Encoded data is invalid");
|
2024-04-19 14:23:16 -06:00
|
|
|
async_did_fail_to_decode_image(image_id, "Encoded data is invalid"_string);
|
|
|
|
return image_id;
|
2021-01-29 22:30:48 +01:00
|
|
|
}
|
|
|
|
|
2024-04-19 14:23:16 -06:00
|
|
|
m_pending_jobs.set(image_id, make_decode_image_job(image_id, encoded_buffer, ideal_size, mime_type));
|
|
|
|
|
|
|
|
Core::deferred_invoke([strong_this = NonnullRefPtr(*this), image_id]() {
|
|
|
|
if (auto job = strong_this->m_pending_jobs.take(image_id); job.has_value()) {
|
|
|
|
job.value()();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
return image_id;
|
|
|
|
}
|
|
|
|
|
|
|
|
void ConnectionFromClient::cancel_decoding(i64 image_id)
|
|
|
|
{
|
|
|
|
if (auto job = m_pending_jobs.take(image_id); job.has_value()) {
|
|
|
|
async_did_fail_to_decode_image(image_id, "Decoding was canceled"_string);
|
|
|
|
}
|
2020-06-22 21:35:22 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
}
|