| /* |
| * Copyright 2022 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| #include <algorithm> |
| #include <cmath> |
| |
| #include "ultrahdr/ultrahdrcommon.h" |
| #include "ultrahdr/jpegr.h" |
| #include "ultrahdr/jpegrutils.h" |
| |
| #include "image_io/xml/xml_reader.h" |
| #include "image_io/xml/xml_writer.h" |
| #include "image_io/base/message_handler.h" |
| #include "image_io/xml/xml_element_rules.h" |
| #include "image_io/xml/xml_handler.h" |
| #include "image_io/xml/xml_rule.h" |
| |
| using namespace photos_editing_formats::image_io; |
| using namespace std; |
| |
| namespace ultrahdr { |
| /* |
| * Helper function used for generating XMP metadata. |
| * |
| * @param prefix The prefix part of the name. |
| * @param suffix The suffix part of the name. |
| * @return A name of the form "prefix:suffix". |
| */ |
| static inline string Name(const string& prefix, const string& suffix) { |
| std::stringstream ss; |
| ss << prefix << ":" << suffix; |
| return ss.str(); |
| } |
| |
| DataStruct::DataStruct(size_t s) { |
| data = malloc(s); |
| length = s; |
| memset(data, 0, s); |
| writePos = 0; |
| } |
| |
| DataStruct::~DataStruct() { |
| if (data != nullptr) { |
| free(data); |
| } |
| } |
| |
| void* DataStruct::getData() { return data; } |
| |
| size_t DataStruct::getLength() { return length; } |
| |
| size_t DataStruct::getBytesWritten() { return writePos; } |
| |
| bool DataStruct::write8(uint8_t value) { |
| uint8_t v = value; |
| return write(&v, 1); |
| } |
| |
| bool DataStruct::write16(uint16_t value) { |
| uint16_t v = value; |
| return write(&v, 2); |
| } |
| |
| bool DataStruct::write32(uint32_t value) { |
| uint32_t v = value; |
| return write(&v, 4); |
| } |
| |
| bool DataStruct::write(const void* src, size_t size) { |
| if (writePos + size > length) { |
| ALOGE("Writing out of boundary: write position: %zd, size: %zd, capacity: %zd", writePos, size, |
| length); |
| return false; |
| } |
| memcpy((uint8_t*)data + writePos, src, size); |
| writePos += size; |
| return true; |
| } |
| |
| /* |
| * Helper function used for writing data to destination. |
| */ |
| uhdr_error_info_t Write(uhdr_compressed_image_t* destination, const void* source, size_t length, |
| size_t& position) { |
| if (position + length > destination->capacity) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_MEM_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, |
| "output buffer to store compressed data is too small: write position: %zd, size: %zd, " |
| "capacity: %zd", |
| position, length, destination->capacity); |
| return status; |
| } |
| |
| memcpy((uint8_t*)destination->data + sizeof(uint8_t) * position, source, length); |
| position += length; |
| return g_no_error; |
| } |
| |
| // Extremely simple XML Handler - just searches for interesting elements |
| class XMPXmlHandler : public XmlHandler { |
| public: |
| XMPXmlHandler() : XmlHandler() { |
| state = NotStrarted; |
| versionFound = false; |
| minContentBoostFound = false; |
| maxContentBoostFound = false; |
| gammaFound = false; |
| offsetSdrFound = false; |
| offsetHdrFound = false; |
| hdrCapacityMinFound = false; |
| hdrCapacityMaxFound = false; |
| baseRenditionIsHdrFound = false; |
| } |
| |
| enum ParseState { NotStrarted, Started, Done }; |
| |
| virtual DataMatchResult StartElement(const XmlTokenContext& context) { |
| string val; |
| if (context.BuildTokenValue(&val)) { |
| if (!val.compare(containerName)) { |
| state = Started; |
| } else { |
| if (state != Done) { |
| state = NotStrarted; |
| } |
| } |
| } |
| return context.GetResult(); |
| } |
| |
| virtual DataMatchResult FinishElement(const XmlTokenContext& context) { |
| if (state == Started) { |
| state = Done; |
| lastAttributeName = ""; |
| } |
| return context.GetResult(); |
| } |
| |
| virtual DataMatchResult AttributeName(const XmlTokenContext& context) { |
| string val; |
| if (state == Started) { |
| if (context.BuildTokenValue(&val)) { |
| if (!val.compare(versionAttrName)) { |
| lastAttributeName = versionAttrName; |
| } else if (!val.compare(maxContentBoostAttrName)) { |
| lastAttributeName = maxContentBoostAttrName; |
| } else if (!val.compare(minContentBoostAttrName)) { |
| lastAttributeName = minContentBoostAttrName; |
| } else if (!val.compare(gammaAttrName)) { |
| lastAttributeName = gammaAttrName; |
| } else if (!val.compare(offsetSdrAttrName)) { |
| lastAttributeName = offsetSdrAttrName; |
| } else if (!val.compare(offsetHdrAttrName)) { |
| lastAttributeName = offsetHdrAttrName; |
| } else if (!val.compare(hdrCapacityMinAttrName)) { |
| lastAttributeName = hdrCapacityMinAttrName; |
| } else if (!val.compare(hdrCapacityMaxAttrName)) { |
| lastAttributeName = hdrCapacityMaxAttrName; |
| } else if (!val.compare(baseRenditionIsHdrAttrName)) { |
| lastAttributeName = baseRenditionIsHdrAttrName; |
| } else { |
| lastAttributeName = ""; |
| } |
| } |
| } |
| return context.GetResult(); |
| } |
| |
| virtual DataMatchResult AttributeValue(const XmlTokenContext& context) { |
| string val; |
| if (state == Started) { |
| if (context.BuildTokenValue(&val, true)) { |
| if (!lastAttributeName.compare(versionAttrName)) { |
| versionStr = val; |
| versionFound = true; |
| } else if (!lastAttributeName.compare(maxContentBoostAttrName)) { |
| maxContentBoostStr = val; |
| maxContentBoostFound = true; |
| } else if (!lastAttributeName.compare(minContentBoostAttrName)) { |
| minContentBoostStr = val; |
| minContentBoostFound = true; |
| } else if (!lastAttributeName.compare(gammaAttrName)) { |
| gammaStr = val; |
| gammaFound = true; |
| } else if (!lastAttributeName.compare(offsetSdrAttrName)) { |
| offsetSdrStr = val; |
| offsetSdrFound = true; |
| } else if (!lastAttributeName.compare(offsetHdrAttrName)) { |
| offsetHdrStr = val; |
| offsetHdrFound = true; |
| } else if (!lastAttributeName.compare(hdrCapacityMinAttrName)) { |
| hdrCapacityMinStr = val; |
| hdrCapacityMinFound = true; |
| } else if (!lastAttributeName.compare(hdrCapacityMaxAttrName)) { |
| hdrCapacityMaxStr = val; |
| hdrCapacityMaxFound = true; |
| } else if (!lastAttributeName.compare(baseRenditionIsHdrAttrName)) { |
| baseRenditionIsHdrStr = val; |
| baseRenditionIsHdrFound = true; |
| } |
| } |
| } |
| return context.GetResult(); |
| } |
| |
| bool getVersion(string* version, bool* present) { |
| if (state == Done) { |
| *version = versionStr; |
| *present = versionFound; |
| return true; |
| } else { |
| return false; |
| } |
| } |
| |
| bool getMaxContentBoost(float* max_content_boost, bool* present) { |
| if (state == Done) { |
| *present = maxContentBoostFound; |
| stringstream ss(maxContentBoostStr); |
| float val; |
| if (ss >> val) { |
| *max_content_boost = exp2(val); |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getMinContentBoost(float* min_content_boost, bool* present) { |
| if (state == Done) { |
| *present = minContentBoostFound; |
| stringstream ss(minContentBoostStr); |
| float val; |
| if (ss >> val) { |
| *min_content_boost = exp2(val); |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getGamma(float* gamma, bool* present) { |
| if (state == Done) { |
| *present = gammaFound; |
| stringstream ss(gammaStr); |
| float val; |
| if (ss >> val) { |
| *gamma = val; |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getOffsetSdr(float* offset_sdr, bool* present) { |
| if (state == Done) { |
| *present = offsetSdrFound; |
| stringstream ss(offsetSdrStr); |
| float val; |
| if (ss >> val) { |
| *offset_sdr = val; |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getOffsetHdr(float* offset_hdr, bool* present) { |
| if (state == Done) { |
| *present = offsetHdrFound; |
| stringstream ss(offsetHdrStr); |
| float val; |
| if (ss >> val) { |
| *offset_hdr = val; |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getHdrCapacityMin(float* hdr_capacity_min, bool* present) { |
| if (state == Done) { |
| *present = hdrCapacityMinFound; |
| stringstream ss(hdrCapacityMinStr); |
| float val; |
| if (ss >> val) { |
| *hdr_capacity_min = exp2(val); |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getHdrCapacityMax(float* hdr_capacity_max, bool* present) { |
| if (state == Done) { |
| *present = hdrCapacityMaxFound; |
| stringstream ss(hdrCapacityMaxStr); |
| float val; |
| if (ss >> val) { |
| *hdr_capacity_max = exp2(val); |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| bool getBaseRenditionIsHdr(bool* base_rendition_is_hdr, bool* present) { |
| if (state == Done) { |
| *present = baseRenditionIsHdrFound; |
| if (!baseRenditionIsHdrStr.compare("False")) { |
| *base_rendition_is_hdr = false; |
| return true; |
| } else if (!baseRenditionIsHdrStr.compare("True")) { |
| *base_rendition_is_hdr = true; |
| return true; |
| } else { |
| return false; |
| } |
| } else { |
| return false; |
| } |
| } |
| |
| private: |
| static const string containerName; |
| |
| static const string versionAttrName; |
| string versionStr; |
| bool versionFound; |
| static const string maxContentBoostAttrName; |
| string maxContentBoostStr; |
| bool maxContentBoostFound; |
| static const string minContentBoostAttrName; |
| string minContentBoostStr; |
| bool minContentBoostFound; |
| static const string gammaAttrName; |
| string gammaStr; |
| bool gammaFound; |
| static const string offsetSdrAttrName; |
| string offsetSdrStr; |
| bool offsetSdrFound; |
| static const string offsetHdrAttrName; |
| string offsetHdrStr; |
| bool offsetHdrFound; |
| static const string hdrCapacityMinAttrName; |
| string hdrCapacityMinStr; |
| bool hdrCapacityMinFound; |
| static const string hdrCapacityMaxAttrName; |
| string hdrCapacityMaxStr; |
| bool hdrCapacityMaxFound; |
| static const string baseRenditionIsHdrAttrName; |
| string baseRenditionIsHdrStr; |
| bool baseRenditionIsHdrFound; |
| |
| string lastAttributeName; |
| ParseState state; |
| }; |
| |
| // GContainer XMP constants - URI and namespace prefix |
| const string kContainerUri = "http://ns.google.com/photos/1.0/container/"; |
| const string kContainerPrefix = "Container"; |
| |
| // GContainer XMP constants - element and attribute names |
| const string kConDirectory = Name(kContainerPrefix, "Directory"); |
| const string kConItem = Name(kContainerPrefix, "Item"); |
| |
| // GContainer XMP constants - names for XMP handlers |
| const string XMPXmlHandler::containerName = "rdf:Description"; |
| // Item XMP constants - URI and namespace prefix |
| const string kItemUri = "http://ns.google.com/photos/1.0/container/item/"; |
| const string kItemPrefix = "Item"; |
| |
| // Item XMP constants - element and attribute names |
| const string kItemLength = Name(kItemPrefix, "Length"); |
| const string kItemMime = Name(kItemPrefix, "Mime"); |
| const string kItemSemantic = Name(kItemPrefix, "Semantic"); |
| |
| // Item XMP constants - element and attribute values |
| const string kSemanticPrimary = "Primary"; |
| const string kSemanticGainMap = "GainMap"; |
| const string kMimeImageJpeg = "image/jpeg"; |
| |
| // GainMap XMP constants - URI and namespace prefix |
| const string kGainMapUri = "http://ns.adobe.com/hdr-gain-map/1.0/"; |
| const string kGainMapPrefix = "hdrgm"; |
| |
| // GainMap XMP constants - element and attribute names |
| const string kMapVersion = Name(kGainMapPrefix, "Version"); |
| const string kMapGainMapMin = Name(kGainMapPrefix, "GainMapMin"); |
| const string kMapGainMapMax = Name(kGainMapPrefix, "GainMapMax"); |
| const string kMapGamma = Name(kGainMapPrefix, "Gamma"); |
| const string kMapOffsetSdr = Name(kGainMapPrefix, "OffsetSDR"); |
| const string kMapOffsetHdr = Name(kGainMapPrefix, "OffsetHDR"); |
| const string kMapHDRCapacityMin = Name(kGainMapPrefix, "HDRCapacityMin"); |
| const string kMapHDRCapacityMax = Name(kGainMapPrefix, "HDRCapacityMax"); |
| const string kMapBaseRenditionIsHDR = Name(kGainMapPrefix, "BaseRenditionIsHDR"); |
| |
| // GainMap XMP constants - names for XMP handlers |
| const string XMPXmlHandler::versionAttrName = kMapVersion; |
| const string XMPXmlHandler::minContentBoostAttrName = kMapGainMapMin; |
| const string XMPXmlHandler::maxContentBoostAttrName = kMapGainMapMax; |
| const string XMPXmlHandler::gammaAttrName = kMapGamma; |
| const string XMPXmlHandler::offsetSdrAttrName = kMapOffsetSdr; |
| const string XMPXmlHandler::offsetHdrAttrName = kMapOffsetHdr; |
| const string XMPXmlHandler::hdrCapacityMinAttrName = kMapHDRCapacityMin; |
| const string XMPXmlHandler::hdrCapacityMaxAttrName = kMapHDRCapacityMax; |
| const string XMPXmlHandler::baseRenditionIsHdrAttrName = kMapBaseRenditionIsHDR; |
| |
| uhdr_error_info_t getMetadataFromXMP(uint8_t* xmp_data, size_t xmp_size, |
| uhdr_gainmap_metadata_ext_t* metadata) { |
| string nameSpace = "http://ns.adobe.com/xap/1.0/\0"; |
| |
| if (xmp_size < nameSpace.size() + 2) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, |
| "size of xmp block is expected to be atleast %zd bytes, received only %zd bytes", |
| nameSpace.size() + 2, xmp_size); |
| return status; |
| } |
| |
| if (strncmp(reinterpret_cast<char*>(xmp_data), nameSpace.c_str(), nameSpace.size())) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, |
| "mismatch in namespace of xmp block. Expected %s, Got %.*s", nameSpace.c_str(), |
| (int)nameSpace.size(), reinterpret_cast<char*>(xmp_data)); |
| return status; |
| } |
| |
| // Position the pointers to the start of XMP XML portion |
| xmp_data += nameSpace.size() + 1; |
| xmp_size -= nameSpace.size() + 1; |
| XMPXmlHandler handler; |
| |
| // xml parser fails to parse packet header, wrapper. remove them before handing the data to |
| // parser. if there is no packet header, do nothing otherwise go to the position of '<' without |
| // '?' after it. |
| size_t offset = 0; |
| for (size_t i = 0; i < xmp_size - 1; ++i) { |
| if (xmp_data[i] == '<') { |
| if (xmp_data[i + 1] != '?') { |
| offset = i; |
| break; |
| } |
| } |
| } |
| xmp_data += offset; |
| xmp_size -= offset; |
| |
| // If there is no packet wrapper, do nothing other wise go to the position of last '>' without '?' |
| // before it. |
| offset = 0; |
| for (size_t i = xmp_size - 1; i >= 1; --i) { |
| if (xmp_data[i] == '>') { |
| if (xmp_data[i - 1] != '?') { |
| offset = xmp_size - (i + 1); |
| break; |
| } |
| } |
| } |
| xmp_size -= offset; |
| |
| // remove padding |
| while (xmp_data[xmp_size - 1] != '>' && xmp_size > 1) { |
| xmp_size--; |
| } |
| |
| string str(reinterpret_cast<const char*>(xmp_data), xmp_size); |
| MessageHandler msg_handler; |
| unique_ptr<XmlRule> rule(new XmlElementRule); |
| XmlReader reader(&handler, &msg_handler); |
| reader.StartParse(std::move(rule)); |
| reader.Parse(str); |
| reader.FinishParse(); |
| if (reader.HasErrors()) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_UNKNOWN_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parser returned with error"); |
| return status; |
| } |
| |
| // Apply default values to any not-present fields, except for Version, |
| // maxContentBoost, and hdrCapacityMax, which are required. Return false if |
| // we encounter a present field that couldn't be parsed, since this |
| // indicates it is invalid (eg. string where there should be a float). |
| bool present = false; |
| if (!handler.getVersion(&metadata->version, &present) || !present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, could not find attribute %s", |
| kMapVersion.c_str()); |
| return status; |
| } |
| if (!handler.getMaxContentBoost(&metadata->max_content_boost, &present) || !present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, could not find attribute %s", |
| kMapGainMapMax.c_str()); |
| return status; |
| } |
| if (!handler.getHdrCapacityMax(&metadata->hdr_capacity_max, &present) || !present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, could not find attribute %s", |
| kMapHDRCapacityMax.c_str()); |
| return status; |
| } |
| if (!handler.getMinContentBoost(&metadata->min_content_boost, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapGainMapMin.c_str()); |
| return status; |
| } |
| metadata->min_content_boost = 1.0f; |
| } |
| if (!handler.getGamma(&metadata->gamma, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapGamma.c_str()); |
| return status; |
| } |
| metadata->gamma = 1.0f; |
| } |
| if (!handler.getOffsetSdr(&metadata->offset_sdr, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapOffsetSdr.c_str()); |
| return status; |
| } |
| metadata->offset_sdr = 1.0f / 64.0f; |
| } |
| if (!handler.getOffsetHdr(&metadata->offset_hdr, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapOffsetHdr.c_str()); |
| return status; |
| } |
| metadata->offset_hdr = 1.0f / 64.0f; |
| } |
| if (!handler.getHdrCapacityMin(&metadata->hdr_capacity_min, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapHDRCapacityMin.c_str()); |
| return status; |
| } |
| metadata->hdr_capacity_min = 1.0f; |
| } |
| |
| bool base_rendition_is_hdr; |
| if (!handler.getBaseRenditionIsHdr(&base_rendition_is_hdr, &present)) { |
| if (present) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "xml parse error, unable to parse attribute %s", |
| kMapBaseRenditionIsHDR.c_str()); |
| return status; |
| } |
| base_rendition_is_hdr = false; |
| } |
| if (base_rendition_is_hdr) { |
| uhdr_error_info_t status; |
| status.error_code = UHDR_CODEC_ERROR; |
| status.has_detail = 1; |
| snprintf(status.detail, sizeof status.detail, "hdr intent as base rendition is not supported"); |
| return status; |
| } |
| |
| return g_no_error; |
| } |
| |
| string generateXmpForPrimaryImage(size_t secondary_image_length, |
| uhdr_gainmap_metadata_ext_t& metadata) { |
| const vector<string> kConDirSeq({kConDirectory, string("rdf:Seq")}); |
| const vector<string> kLiItem({string("rdf:li"), kConItem}); |
| |
| std::stringstream ss; |
| photos_editing_formats::image_io::XmlWriter writer(ss); |
| writer.StartWritingElement("x:xmpmeta"); |
| writer.WriteXmlns("x", "adobe:ns:meta/"); |
| writer.WriteAttributeNameAndValue("x:xmptk", "Adobe XMP Core 5.1.2"); |
| writer.StartWritingElement("rdf:RDF"); |
| writer.WriteXmlns("rdf", "http://www.w3.org/1999/02/22-rdf-syntax-ns#"); |
| writer.StartWritingElement("rdf:Description"); |
| writer.WriteXmlns(kContainerPrefix, kContainerUri); |
| writer.WriteXmlns(kItemPrefix, kItemUri); |
| writer.WriteXmlns(kGainMapPrefix, kGainMapUri); |
| writer.WriteAttributeNameAndValue(kMapVersion, metadata.version); |
| |
| writer.StartWritingElements(kConDirSeq); |
| |
| size_t item_depth = writer.StartWritingElement("rdf:li"); |
| writer.WriteAttributeNameAndValue("rdf:parseType", "Resource"); |
| writer.StartWritingElement(kConItem); |
| writer.WriteAttributeNameAndValue(kItemSemantic, kSemanticPrimary); |
| writer.WriteAttributeNameAndValue(kItemMime, kMimeImageJpeg); |
| writer.FinishWritingElementsToDepth(item_depth); |
| |
| writer.StartWritingElement("rdf:li"); |
| writer.WriteAttributeNameAndValue("rdf:parseType", "Resource"); |
| writer.StartWritingElement(kConItem); |
| writer.WriteAttributeNameAndValue(kItemSemantic, kSemanticGainMap); |
| writer.WriteAttributeNameAndValue(kItemMime, kMimeImageJpeg); |
| writer.WriteAttributeNameAndValue(kItemLength, secondary_image_length); |
| |
| writer.FinishWriting(); |
| |
| return ss.str(); |
| } |
| |
| string generateXmpForSecondaryImage(uhdr_gainmap_metadata_ext_t& metadata) { |
| const vector<string> kConDirSeq({kConDirectory, string("rdf:Seq")}); |
| |
| std::stringstream ss; |
| photos_editing_formats::image_io::XmlWriter writer(ss); |
| writer.StartWritingElement("x:xmpmeta"); |
| writer.WriteXmlns("x", "adobe:ns:meta/"); |
| writer.WriteAttributeNameAndValue("x:xmptk", "Adobe XMP Core 5.1.2"); |
| writer.StartWritingElement("rdf:RDF"); |
| writer.WriteXmlns("rdf", "http://www.w3.org/1999/02/22-rdf-syntax-ns#"); |
| writer.StartWritingElement("rdf:Description"); |
| writer.WriteXmlns(kGainMapPrefix, kGainMapUri); |
| writer.WriteAttributeNameAndValue(kMapVersion, metadata.version); |
| writer.WriteAttributeNameAndValue(kMapGainMapMin, log2(metadata.min_content_boost)); |
| writer.WriteAttributeNameAndValue(kMapGainMapMax, log2(metadata.max_content_boost)); |
| writer.WriteAttributeNameAndValue(kMapGamma, metadata.gamma); |
| writer.WriteAttributeNameAndValue(kMapOffsetSdr, metadata.offset_sdr); |
| writer.WriteAttributeNameAndValue(kMapOffsetHdr, metadata.offset_hdr); |
| writer.WriteAttributeNameAndValue(kMapHDRCapacityMin, log2(metadata.hdr_capacity_min)); |
| writer.WriteAttributeNameAndValue(kMapHDRCapacityMax, log2(metadata.hdr_capacity_max)); |
| writer.WriteAttributeNameAndValue(kMapBaseRenditionIsHDR, "False"); |
| writer.FinishWriting(); |
| |
| return ss.str(); |
| } |
| |
| } // namespace ultrahdr |