ladybird/Userland/Libraries/LibGfx/ICC/BinaryWriter.cpp
Nico Weber 4e72a35398 LibGfx: Add scaffolding for writing tag data
This doesn't deduplicate identical TagDatas yet.

It also doesn't implement actual serialization of TagData yet.
2023-02-19 00:01:44 +01:00

134 lines
4.9 KiB
C++

/*
* Copyright (c) 2023, Nico Weber <thakis@chromium.org>
*
* SPDX-License-Identifier: BSD-2-Clause
*/
#include <LibGfx/ICC/BinaryFormat.h>
#include <LibGfx/ICC/BinaryWriter.h>
#include <LibGfx/ICC/Profile.h>
#include <time.h>
#pragma GCC diagnostic ignored "-Warray-bounds"
namespace Gfx::ICC {
static ErrorOr<ByteBuffer> encode_tag_data(TagData const& tag_data)
{
(void)tag_data;
return ByteBuffer {};
}
static ErrorOr<Vector<ByteBuffer>> encode_tag_datas(Profile const& profile)
{
Vector<ByteBuffer> tag_data_bytes;
// FIXME: If two tags refer to the same TagData object, write it just once to the output.
TRY(tag_data_bytes.try_resize(profile.tag_count()));
size_t i = 0;
profile.for_each_tag([&](auto, auto tag_data) {
// FIXME: Come up with a way to allow TRY instead of MUST here.
tag_data_bytes[i++] = MUST(encode_tag_data(tag_data));
});
return tag_data_bytes;
}
static ErrorOr<void> encode_tag_table(ByteBuffer& bytes, Profile const& profile, Vector<size_t> const& offsets, Vector<ByteBuffer> const& tag_data_bytes)
{
VERIFY(bytes.size() >= sizeof(ICCHeader) + sizeof(u32) + profile.tag_count() * sizeof(TagTableEntry));
*bit_cast<BigEndian<u32>*>(bytes.data() + sizeof(ICCHeader)) = profile.tag_count();
TagTableEntry* tag_table_entries = bit_cast<TagTableEntry*>(bytes.data() + sizeof(ICCHeader) + sizeof(u32));
int i = 0;
profile.for_each_tag([&](auto tag_signature, auto) {
tag_table_entries[i].tag_signature = tag_signature;
tag_table_entries[i].offset_to_beginning_of_tag_data_element = offsets[i];
tag_table_entries[i].size_of_tag_data_element = tag_data_bytes[i].size();
++i;
});
return {};
}
static ErrorOr<void> encode_header(ByteBuffer& bytes, Profile const& profile)
{
VERIFY(bytes.size() >= sizeof(ICCHeader));
auto& raw_header = *bit_cast<ICCHeader*>(bytes.data());
raw_header.profile_size = bytes.size();
raw_header.preferred_cmm_type = profile.preferred_cmm_type().value_or(PreferredCMMType { 0 });
raw_header.profile_version_major = profile.version().major_version();
raw_header.profile_version_minor_bugfix = profile.version().minor_and_bugfix_version();
raw_header.profile_version_zero = 0;
raw_header.profile_device_class = profile.device_class();
raw_header.data_color_space = profile.data_color_space();
raw_header.profile_connection_space = profile.connection_space();
time_t profile_timestamp = profile.creation_timestamp();
struct tm tm;
if (!gmtime_r(&profile_timestamp, &tm))
return Error::from_errno(errno);
raw_header.profile_creation_time.year = tm.tm_year + 1900;
raw_header.profile_creation_time.month = tm.tm_mon + 1;
raw_header.profile_creation_time.day = tm.tm_mday;
raw_header.profile_creation_time.hours = tm.tm_hour;
raw_header.profile_creation_time.minutes = tm.tm_min;
raw_header.profile_creation_time.seconds = tm.tm_sec;
raw_header.profile_file_signature = ProfileFileSignature;
raw_header.primary_platform = profile.primary_platform().value_or(PrimaryPlatform { 0 });
raw_header.profile_flags = profile.flags().bits();
raw_header.device_manufacturer = profile.device_manufacturer().value_or(DeviceManufacturer { 0 });
raw_header.device_model = profile.device_model().value_or(DeviceModel { 0 });
raw_header.device_attributes = profile.device_attributes().bits();
raw_header.rendering_intent = profile.rendering_intent();
raw_header.pcs_illuminant = profile.pcs_illuminant();
raw_header.profile_creator = profile.creator().value_or(Creator { 0 });
memset(raw_header.reserved, 0, sizeof(raw_header.reserved));
auto id = Profile::compute_id(bytes);
static_assert(sizeof(id.data) == sizeof(raw_header.profile_id));
memcpy(raw_header.profile_id, id.data, sizeof(id.data));
return {};
}
ErrorOr<ByteBuffer> encode(Profile const& profile)
{
// Valid profiles always have tags. Profile only represents valid profiles.
VERIFY(profile.tag_count() > 0);
Vector<ByteBuffer> tag_data_bytes = TRY(encode_tag_datas(profile));
size_t tag_table_size = sizeof(u32) + profile.tag_count() * sizeof(TagTableEntry);
size_t offset = sizeof(ICCHeader) + tag_table_size;
Vector<size_t> offsets;
for (auto const& bytes : tag_data_bytes) {
TRY(offsets.try_append(offset));
offset += align_up_to(bytes.size(), 4);
}
// Omit padding after last element.
// FIXME: Is that correct?
size_t total_size = offsets.last() + tag_data_bytes.last().size();
// Leave enough room for the profile header and the tag table count.
auto bytes = TRY(ByteBuffer::create_zeroed(total_size));
for (size_t i = 0; i < tag_data_bytes.size(); ++i)
memcpy(bytes.data() + offsets[i], tag_data_bytes[i].data(), tag_data_bytes[i].size());
TRY(encode_tag_table(bytes, profile, offsets, tag_data_bytes));
TRY(encode_header(bytes, profile));
return bytes;
}
}