Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

evc: initial work for EVC #1352

Draft
wants to merge 9 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions libheif/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -87,6 +87,10 @@ set(libheif_sources
codecs/avc_boxes.cc
codecs/avc_dec.h
codecs/avc_dec.cc
codecs/evc_boxes.cc
codecs/evc_boxes.h
codecs/evc_dec.h
codecs/evc_dec.cc
image-items/mask_image.h
image-items/mask_image.cc
image-items/image_item.h
Expand Down
8 changes: 8 additions & 0 deletions libheif/box.cc
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,7 @@
#include "codecs/avc_boxes.h"
#include "codecs/avif_boxes.h"
#include "image-items/tiled.h"
#include "codecs/evc_boxes.h"

#include <iomanip>
#include <utility>
Expand Down Expand Up @@ -691,6 +692,13 @@ Error Box::read(BitstreamRange& range, std::shared_ptr<Box>* result, const heif_
break;

#if HEIF_ENABLE_EXPERIMENTAL_FEATURES
// --- EVC (ISO/IEC 23094-1)

case fourcc("evcC"):
box = std::make_shared<Box_evcC>();
break;

// --- Low overhead tiling
case fourcc("tilC"):
box = std::make_shared<Box_tilC>();
break;
Expand Down
7 changes: 6 additions & 1 deletion libheif/codecs/decoder.cc
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,8 @@
#include "jpeg_boxes.h"
#include "jpeg2000_boxes.h"
#include "codecs/uncompressed/unc_dec.h"

#include "codecs/evc_dec.h"
#include "evc_boxes.h"

void DataExtent::set_from_image_item(std::shared_ptr<HeifFile> file, heif_item_id item)
{
Expand Down Expand Up @@ -134,6 +135,10 @@ std::shared_ptr<Decoder> Decoder::alloc_for_infe_type(const ImageItem* item)
return std::make_shared<Decoder_uncompressed>(uncC,cmpd);
}
#endif
case fourcc("evc1"): {
auto evcC = item->get_property<Box_evcC>();
return std::make_shared<Decoder_EVC>(evcC);
}
case fourcc("mski"): {
return nullptr; // do we need a decoder for this?
}
Expand Down
226 changes: 226 additions & 0 deletions libheif/codecs/evc_boxes.cc
Original file line number Diff line number Diff line change
@@ -0,0 +1,226 @@
/*
* HEIF EVC codec.
* Copyright (c) 2024 Brad Hards <[email protected]>
*
* This file is part of libheif.
*
* libheif is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation, either version 3 of
* the License, or (at your option) any later version.
*
* libheif is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with libheif. If not, see <http://www.gnu.org/licenses/>.
*/

#include "evc_boxes.h"
#include "bitstream.h"
#include "error.h"
#include "file.h"

#include <iomanip>
#include <string>
#include <utility>
#include <vector>
#include <libheif/api_structs.h>



Error Box_evcC::parse(BitstreamRange& range, const heif_security_limits* limits)
{
m_configuration.configurationVersion = range.read8();
m_configuration.profile_idc = range.read8();
m_configuration.level_idc = range.read8();
m_configuration.toolset_idc_h = range.read32();
m_configuration.toolset_idc_l = range.read32();
uint8_t b = range.read8();
m_configuration.chroma_format_idc = (b >> 6) & 0b11;
uint8_t bit_depth_luma_minus8 = (b >> 3) & 0b111;
m_configuration.bit_depth_luma = bit_depth_luma_minus8 + 8;
uint8_t bit_depth_chroma_minus8 = b & 0b111;
m_configuration.bit_depth_chroma = bit_depth_chroma_minus8 + 8;
m_configuration.pic_width_in_luma_samples = range.read16();
m_configuration.pic_height_in_luma_samples = range.read16();
b = range.read8();
uint8_t lengthSizeMinus1 = b & 0b11;
m_configuration.lengthSize = lengthSizeMinus1 + 1;
uint8_t num_of_arrays = range.read8();
for (uint8_t j = 0; j < num_of_arrays && !range.error(); j++) {
NalArray array;
b = range.read8();
array.array_completeness = ((b & 0x80) == 0x80);
array.NAL_unit_type = (b & 0b00111111);
uint16_t num_nalus = range.read16();
for (int i = 0; i < num_nalus && !range.error(); i++) {
uint16_t nal_unit_length = range.read16();
if (nal_unit_length == 0) {
// Ignore empty NAL units.
continue;
}
std::vector<uint8_t> nal_unit;
if (range.prepare_read(nal_unit_length)) {
nal_unit.resize(nal_unit_length);
bool success = range.get_istream()->read((char*) nal_unit.data(), nal_unit_length);
if (!success) {
return Error{heif_error_Invalid_input, heif_suberror_End_of_data, "error while reading evcC box"};
}
}

array.nal_units.push_back(std::move(nal_unit));
}
m_nal_array.push_back(array);
}

return range.get_error();
}


std::string Box_evcC::dump(Indent& indent) const
{
std::ostringstream sstr;
sstr << Box::dump(indent);
// TODO: decode more of this
sstr << indent << "configurationVersion: " << ((int)m_configuration.configurationVersion) << "\n";
sstr << indent << "profile_idc: " << ((int)m_configuration.profile_idc)
<< " (" << get_profile_as_text() << ")" << "\n";
sstr << indent << "level_idc: " << ((int)m_configuration.level_idc) << "\n";
sstr << indent << "toolset_idc_h: " << m_configuration.toolset_idc_h << "\n";
sstr << indent << "toolset_idc_l: " << m_configuration.toolset_idc_l << "\n";
sstr << indent << "chroma_format_idc: " << ((int)m_configuration.chroma_format_idc)
<< " (" << get_chroma_format_as_text() << ")\n";
sstr << indent << "bit_depth_luma: " << ((int)m_configuration.bit_depth_luma) << "\n";
sstr << indent << "bit_depth_chroma: " << ((int)m_configuration.bit_depth_chroma) << "\n";
sstr << indent << "pic_width_in_luma_samples: " << m_configuration.pic_width_in_luma_samples << "\n";
sstr << indent << "pic_height_in_luma_samples: " << m_configuration.pic_height_in_luma_samples << "\n";
sstr << indent << "length_size: " << ((int)m_configuration.lengthSize) << "\n";
for (const auto &array : m_nal_array)
{
sstr << indent << "<array>\n";

indent++;
sstr << indent << "array_completeness: " << (array.array_completeness ? "true" : "false") << "\n"
<< indent << "NAL_unit_type: " << ((int) array.NAL_unit_type) << " ("
<< get_NAL_unit_type_as_text(array.NAL_unit_type) << ")" << "\n";

for (const auto& unit : array.nal_units) {
sstr << indent;
for (uint8_t b : unit) {
sstr << std::setfill('0') << std::setw(2) << std::hex << ((int) b) << " ";
}
sstr << "\n";
sstr << std::dec;
}

indent--;
}
return sstr.str();
}

std::string Box_evcC::get_profile_as_text() const
{
switch (m_configuration.profile_idc)
{
case 0:
return "Baseline";
case 1:
return "Main";
case 2:
return "Baseline Still";
case 3:
return "Main Still";
default:
return std::string("Unknown");
}
}

std::string Box_evcC::get_chroma_format_as_text() const
{
switch (m_configuration.chroma_format_idc)
{
case CHROMA_FORMAT_MONOCHROME:
return std::string("Monochrome");
case CHROMA_FORMAT_420:
return std::string("4:2:0");
case CHROMA_FORMAT_422:
return std::string("4:2:2");
case CHROMA_FORMAT_444:
return std::string("4:4:4");
default:
return std::string("Invalid");
}
}

std::string Box_evcC::get_NAL_unit_type_as_text(uint8_t nal_unit_type) const
{
switch (nal_unit_type)
{
case 0:
return "NONIDR_NUT";
case 1:
return "IDR_NUT";
case 24:
return "SPS_NUT";
case 25:
return "PPS_NUT";
case 26:
return "APS_NUT";
case 27:
return "FD_NUT";
case 28:
return "SEI_NUT";
default:
return std::string("Unknown");
}
}

Error Box_evcC::write(StreamWriter& writer) const
{
size_t box_start = reserve_box_header_space(writer);

writer.write8(m_configuration.configurationVersion);
writer.write8(m_configuration.profile_idc);
writer.write8(m_configuration.level_idc);
writer.write32(m_configuration.toolset_idc_h);
writer.write32(m_configuration.toolset_idc_l);
uint8_t chroma_format_idc_bits = (uint8_t)((m_configuration.chroma_format_idc & 0b11) << 6);
uint8_t bit_depth_luma_minus8_bits = (uint8_t)(((m_configuration.bit_depth_luma - 8) & 0b111) << 3);
uint8_t bit_depth_chroma_minus8_bits = (uint8_t)((m_configuration.bit_depth_chroma - 8) & 0b111);
writer.write8(chroma_format_idc_bits | bit_depth_luma_minus8_bits | bit_depth_chroma_minus8_bits);
writer.write16(m_configuration.pic_width_in_luma_samples);
writer.write16(m_configuration.pic_height_in_luma_samples);
writer.write8(m_configuration.lengthSize - 1);
writer.write8((uint8_t)m_nal_array.size());
for (const NalArray& array : m_nal_array) {

writer.write8((uint8_t) ((array.array_completeness? 0x80: 0x00) |
(array.NAL_unit_type & 0b00111111)));

writer.write16((uint16_t)array.nal_units.size());

for (const std::vector<uint8_t>& nal_unit : array.nal_units) {
writer.write16((uint16_t) nal_unit.size());
writer.write(nal_unit);
}
}
prepend_header(writer, box_start);

return Error::Ok;
}

void Box_evcC::get_header_nals(std::vector<uint8_t>& data) const
{
for (const auto& array : m_nal_array) {
for (const auto& nalu : array.nal_units) {
data.push_back((nalu.size() >> 24) & 0xFF);
data.push_back((nalu.size() >> 16) & 0xFF);
data.push_back((nalu.size() >> 8) & 0xFF);
data.push_back((nalu.size() >> 0) & 0xFF);
data.insert(data.end(), nalu.begin(), nalu.end());
}
}
}
90 changes: 90 additions & 0 deletions libheif/codecs/evc_boxes.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,90 @@
/*
* HEIF EVC codec.
* Copyright (c) 2024 Brad Hards <[email protected]>
*
* This file is part of libheif.
*
* libheif is free software: you can redistribute it and/or modify
* it under the terms of the GNU Lesser General Public License as
* published by the Free Software Foundation, either version 3 of
* the License, or (at your option) any later version.
*
* libheif is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Lesser General Public License for more details.
*
* You should have received a copy of the GNU Lesser General Public License
* along with libheif. If not, see <http://www.gnu.org/licenses/>.
*/

#ifndef HEIF_EVC_BOXES_H
#define HEIF_EVC_BOXES_H

#include "box.h"
#include "error.h"
#include <string>
#include <vector>

class Box_evcC : public Box {
public:
Box_evcC() { set_short_type(fourcc("evcC")); }

bool is_essential() const override { return true; }

static const uint8_t CHROMA_FORMAT_MONOCHROME = 0;
static const uint8_t CHROMA_FORMAT_420 = 1;
static const uint8_t CHROMA_FORMAT_422 = 2;
static const uint8_t CHROMA_FORMAT_444 = 3;

struct configuration {
uint8_t configurationVersion = 1;
uint8_t profile_idc;
uint8_t level_idc;
uint32_t toolset_idc_h;
uint32_t toolset_idc_l;
uint8_t chroma_format_idc;
uint8_t bit_depth_luma;
uint8_t bit_depth_chroma;
uint16_t pic_width_in_luma_samples;
uint16_t pic_height_in_luma_samples;
uint8_t lengthSize = 0;
};

void set_configuration(const configuration& config)
{
m_configuration = config;
}

const configuration& get_configuration() const
{
return m_configuration;
}

std::string dump(Indent &) const override;

Error write(StreamWriter &writer) const override;

void get_header_nals(std::vector<uint8_t>& data) const;

protected:
Error parse(BitstreamRange &range, const heif_security_limits* limits) override;

private:
configuration m_configuration;
struct NalArray
{
bool array_completeness;
uint8_t NAL_unit_type;

std::vector<std::vector<uint8_t> > nal_units;
};

std::vector<NalArray> m_nal_array;

std::string get_profile_as_text() const;
std::string get_chroma_format_as_text() const;
std::string get_NAL_unit_type_as_text(uint8_t nal_unit_type) const;
};

#endif
Loading
Loading