9#include "scanlineconverter_p.h"
15#include <QImageReader>
23#ifndef JP2_MAX_IMAGE_WIDTH
24#define JP2_MAX_IMAGE_WIDTH 300000
26#ifndef JP2_MAX_IMAGE_HEIGHT
27#define JP2_MAX_IMAGE_HEIGHT JP2_MAX_IMAGE_WIDTH
33#ifndef JP2_MAX_IMAGE_PIXELS
34#define JP2_MAX_IMAGE_PIXELS std::numeric_limits<qint32>::max()
45#define JP2_SUBTYPE QByteArrayLiteral("JP2")
46#define J2K_SUBTYPE QByteArrayLiteral("J2K")
48static void error_callback(
const char *msg,
void *client_data)
54static void warning_callback(
const char *msg,
void *client_data)
60static void info_callback(
const char *msg,
void *client_data)
66static OPJ_SIZE_T jp2_read(
void *p_buffer, OPJ_SIZE_T p_nb_bytes,
void *p_user_data)
70 return OPJ_SIZE_T(-1);
72 return OPJ_SIZE_T(dev->read((
char*)p_buffer, (qint64)p_nb_bytes));
75static OPJ_SIZE_T jp2_write(
void *p_buffer, OPJ_SIZE_T p_nb_bytes,
void *p_user_data)
79 return OPJ_SIZE_T(-1);
81 return OPJ_SIZE_T(dev->write((
char*)p_buffer, (qint64)p_nb_bytes));
84static OPJ_BOOL jp2_seek(OPJ_OFF_T p_nb_bytes,
void *p_user_data)
90 return dev->seek(p_nb_bytes) ? OPJ_TRUE : OPJ_FALSE;
93static OPJ_OFF_T jp2_skip(OPJ_OFF_T p_nb_bytes,
void *p_user_data)
99 if (dev->seek(dev->pos() + p_nb_bytes)) {
105class JP2HandlerPrivate
109 : m_jp2_stream(nullptr)
110 , m_jp2_image(nullptr)
112 , m_jp2_codec(nullptr)
114 , m_subtype(JP2_SUBTYPE)
117 if (sver.size() == 3) {
119 auto v1 = sver.at(0).toInt(&ok1);
120 auto v2 = sver.at(1).toInt(&ok2);
121 auto v3 = sver.at(2).toInt(&ok3);
122 if (ok1 && ok2 && ok3)
123 m_jp2_version = QT_VERSION_CHECK(v1, v2, v3);
129 opj_image_destroy(m_jp2_image);
130 m_jp2_image =
nullptr;
133 opj_stream_destroy(m_jp2_stream);
134 m_jp2_stream =
nullptr;
137 opj_destroy_codec(m_jp2_codec);
138 m_jp2_codec =
nullptr;
147 OPJ_CODEC_FORMAT detectDecoderFormat(QIODevice *device)
const
149 auto ba = device->
peek(32);
153 return OPJ_CODEC_JP2;
156 return OPJ_CODEC_J2K;
158 return OPJ_CODEC_UNKNOWN;
161 bool createStream(QIODevice *device,
bool read)
163 if (device ==
nullptr) {
166 if (m_jp2_stream ==
nullptr) {
167 m_jp2_stream = opj_stream_default_create(read ? OPJ_TRUE : OPJ_FALSE);
169 if (m_jp2_stream ==
nullptr) {
172 opj_stream_set_user_data(m_jp2_stream, device,
nullptr);
173 opj_stream_set_user_data_length(m_jp2_stream, read ? device->
size() : 0);
174 opj_stream_set_read_function(m_jp2_stream, jp2_read);
175 opj_stream_set_write_function(m_jp2_stream, jp2_write);
176 opj_stream_set_skip_function(m_jp2_stream, jp2_skip);
177 opj_stream_set_seek_function(m_jp2_stream, jp2_seek);
181 bool isImageValid(
const opj_image_t *i)
const
183 return i && i->comps && i->numcomps > 0;
186 void enableThreads(opj_codec_t *codec)
const
188 if (!opj_has_thread_support()) {
189 qInfo() <<
"OpenJPEG doesn't support multi-threading!";
191 qWarning() <<
"Unable to enable multi-threading!";
195 bool createDecoder(QIODevice *device)
200 auto jp2Format = detectDecoderFormat(device);
201 if (jp2Format == OPJ_CODEC_UNKNOWN) {
204 m_jp2_codec = opj_create_decompress(jp2Format);
205 if (m_jp2_codec ==
nullptr) {
208 enableThreads(m_jp2_codec);
213 opj_set_error_handler(m_jp2_codec, error_callback,
nullptr);
217 bool readHeader(QIODevice *device)
219 if (!createStream(device,
true)) {
227 if (!createDecoder(device)) {
231 opj_set_default_decoder_parameters(&m_dparameters);
232 if (!opj_setup_decoder(m_jp2_codec, &m_dparameters)) {
233 qCritical() <<
"Failed to setup JP2 decoder!";
237 if (!opj_read_header(m_jp2_stream, m_jp2_codec, &m_jp2_image)) {
238 qCritical() <<
"Failed to read JP2 header!";
242 return isImageValid(m_jp2_image);
246 bool jp2ToImage(QImage *img)
const
248 Q_ASSERT(img->
depth() == 8 *
sizeof(T) || img->
depth() == 32 *
sizeof(T));
249 for (qint32 c = 0, cc = m_jp2_image->numcomps; c < cc; ++c) {
250 auto cs = cc == 1 ? 1 : 4;
251 auto &&jc = m_jp2_image->comps[c];
252 if (jc.data ==
nullptr)
254 if (qint32(jc.w) != img->
width() || qint32(jc.h) != img->
height())
258 if (std::numeric_limits<T>::is_integer) {
260 if (jc.prec >
sizeof(T) * 8) {
262 divisor = std::max(1,
int(((1ll << jc.prec) - 1) / ((1ll << (
sizeof(T) * 8)) - 1)));
264 for (qint32 y = 0, h = img->
height(); y < h; ++y) {
265 auto ptr =
reinterpret_cast<T *
>(img->
scanLine(y));
266 for (qint32 x = 0, w = img->
width(); x < w; ++x) {
267 qint32 v = jc.data[y * w + x] / divisor;
269 v -= std::numeric_limits<typename std::make_signed<T>::type>::min();
270 *(ptr + x * cs + c) = std::clamp(v, qint32(std::numeric_limits<T>::lowest()), qint32(std::numeric_limits<T>::max()));
274 for (qint32 y = 0, h = img->
height(); y < h; ++y) {
275 auto ptr =
reinterpret_cast<T *
>(img->
scanLine(y));
276 for (qint32 x = 0, w = img->
width(); x < w; ++x) {
277 *(ptr + x * cs + c) = jc.data[y * w + x];
286 void alphaFix(QImage *img)
const
288 if (m_jp2_image->numcomps == 3) {
289 Q_ASSERT(img->
depth() == 32 *
sizeof(T));
290 for (qint32 y = 0, h = img->
height(); y < h; ++y) {
291 auto ptr =
reinterpret_cast<T *
>(img->
scanLine(y));
292 for (qint32 x = 0, w = img->
width(); x < w; ++x) {
293 *(ptr + x * 4 + 3) = std::numeric_limits<T>::is_iec559 ? 1 : std::numeric_limits<T>::max();
299 QImage readImage(QIODevice *device)
301 if (!readHeader(device)) {
305 auto img = imageAlloc(size(), format());
310 if (!opj_decode(m_jp2_codec, m_jp2_stream, m_jp2_image)) {
311 qCritical() <<
"Failed to decoding JP2 image!";
317 if (!jp2ToImage<quint32>(&img))
319 alphaFix<float>(&img);
321 if (!jp2ToImage<quint16>(&img))
323 alphaFix<quint16>(&img);
325 if (!jp2ToImage<quint8>(&img))
327 alphaFix<quint8>(&img);
335 bool checkSizeLimits(qint32 width, qint32 height, qint32 nchannels)
const
337 if (width > JP2_MAX_IMAGE_WIDTH || height > JP2_MAX_IMAGE_HEIGHT || width < 1 || height < 1) {
338 qCritical() <<
"Maximum image size is limited to" << JP2_MAX_IMAGE_WIDTH <<
"x" << JP2_MAX_IMAGE_HEIGHT <<
"pixels";
342 if (qint64(width) * qint64(height) > JP2_MAX_IMAGE_PIXELS) {
343 qCritical() <<
"Maximum image size is limited to" << JP2_MAX_IMAGE_PIXELS <<
"pixels";
349 auto neededBytes = qint64(width) * height * nchannels * 4;
350 if (maxBytes > 0 && neededBytes > maxBytes) {
351 qCritical() <<
"Allocation limit set to" << (maxBytes / 1024 / 1024) <<
"MiB but" << (neededBytes / 1024 / 1024) <<
"MiB are needed!";
358 bool checkSizeLimits(
const QSize &size, qint32 nchannels)
const
360 return checkSizeLimits(size.width(), size.height(), nchannels);
366 if (isImageValid(m_jp2_image)) {
367 auto &&c0 = m_jp2_image->comps[0];
368 auto tmp = QSize(c0.w, c0.h);
369 if (checkSizeLimits(tmp, m_jp2_image->numcomps))
378 if (isImageValid(m_jp2_image)) {
379 auto &&c0 = m_jp2_image->comps[0];
381 for (quint32 c = 1; c < m_jp2_image->numcomps; ++c) {
382 auto &&cc = m_jp2_image->comps[c];
386 auto jp2cs = m_jp2_image->color_space;
387#if QT_VERSION >= QT_VERSION_CHECK(6, 8, 0)
388 if (jp2cs == OPJ_CLRSPC_UNKNOWN || jp2cs == OPJ_CLRSPC_UNSPECIFIED) {
389 auto cs = colorSpace();
390 if (cs.colorModel() == QColorSpace::ColorModel::Cmyk)
391 jp2cs = OPJ_CLRSPC_CMYK;
392 else if (cs.colorModel() == QColorSpace::ColorModel::Rgb)
393 jp2cs = OPJ_CLRSPC_SRGB;
394 else if (cs.colorModel() == QColorSpace::ColorModel::Gray)
395 jp2cs = OPJ_CLRSPC_GRAY;
398 if (jp2cs == OPJ_CLRSPC_UNKNOWN || jp2cs == OPJ_CLRSPC_UNSPECIFIED) {
399 if (m_jp2_image->numcomps == 1)
400 jp2cs = OPJ_CLRSPC_GRAY;
402 jp2cs = OPJ_CLRSPC_SRGB;
406 if (jp2cs == OPJ_CLRSPC_SRGB) {
407 if (m_jp2_image->numcomps == 3 || m_jp2_image->numcomps == 4) {
408 auto hasAlpha = m_jp2_image->numcomps == 4;
418 }
else if (jp2cs == OPJ_CLRSPC_GRAY) {
419 if (m_jp2_image->numcomps == 1) {
425 }
else if (jp2cs == OPJ_CLRSPC_CMYK) {
426 if (m_jp2_image->numcomps == 4) {
427#if QT_VERSION >= QT_VERSION_CHECK(6, 8, 0)
428 if (prec == 8 || prec == 16)
429 fmt = QImage::Format_CMYK8888;
437 QColorSpace colorSpace()
const
441 if (m_jp2_image->icc_profile_buf && m_jp2_image->icc_profile_len > 0) {
445 if (m_jp2_image->color_space == OPJ_CLRSPC_SRGB)
456 bool isSupported()
const
461 QByteArray subType()
const
465 void setSubType(
const QByteArray &type)
470 qint32 quality()
const
474 void setQuality(qint32 quality)
476 m_quality = std::clamp(quality, -1, 100);
483 OPJ_CODEC_FORMAT encoderFormat()
const
485 return subType() == J2K_SUBTYPE ? OPJ_CODEC_J2K : OPJ_CODEC_JP2;
492 qint32 opjVersion()
const
494 return m_jp2_version;
497 bool imageToJp2(
const QImage &image)
501 auto convFormat = image.
format();
502 auto isFloat =
false;
503 auto cs = OPJ_CLRSPC_SRGB;
504 if (opjVersion() >= QT_VERSION_CHECK(2, 5, 4)) {
506 if (!(ics.isValid() && ics.primaries() == QColorSpace::Primaries::SRgb && ics.transferFunction() == QColorSpace::TransferFunction::SRgb)) {
507 cs = OPJ_CLRSPC_UNKNOWN;
517 cs = OPJ_CLRSPC_GRAY;
523 cs = OPJ_CLRSPC_GRAY;
527 cs = OPJ_CLRSPC_SRGB;
534 cs = OPJ_CLRSPC_GRAY;
543 cs = OPJ_CLRSPC_UNSPECIFIED;
563 cs = OPJ_CLRSPC_UNSPECIFIED;
575#if QT_VERSION >= QT_VERSION_CHECK(6, 8, 0)
576 case QImage::Format_CMYK8888:
577 if (opjVersion() >= QT_VERSION_CHECK(2, 5, 3)) {
579 cs = OPJ_CLRSPC_CMYK;
586 if (image.
depth() > 32) {
587 qWarning() <<
"The image is saved losing precision!";
593 if (!checkSizeLimits(image.
size(), ncomp)) {
597 opj_set_default_encoder_parameters(&m_cparameters);
598 m_cparameters.cod_format = encoderFormat();
599 m_cparameters.tile_size_on = 1;
600 m_cparameters.cp_tdx = 1024;
601 m_cparameters.cp_tdy = 1024;
603 if (m_quality > -1 && m_quality < 100) {
604 m_cparameters.irreversible = 1;
605 m_cparameters.tcp_numlayers = 1;
606 m_cparameters.cp_disto_alloc = 1;
607 m_cparameters.tcp_rates[0] = 100.0 - (m_quality < 10 ? m_quality : 10 + (std::log10(m_quality) - 1) * 90);
610 std::unique_ptr<opj_image_cmptparm_t> cmptparm(
new opj_image_cmptparm_t[ncomp]);
611 for (
int i = 0; i < ncomp; ++i) {
612 auto &&p = cmptparm.get() + i;
613 memset(p, 0,
sizeof(opj_image_cmptparm_t));
614 p->dx = m_cparameters.subsampling_dx;
615 p->dy = m_cparameters.subsampling_dy;
616 p->w = image.
width();
624 m_jp2_image = opj_image_create(ncomp, cmptparm.get(), cs);
625 if (m_jp2_image ==
nullptr) {
628 if (
int(m_jp2_image->numcomps) != ncomp) {
631 m_jp2_image->x1 = image.
width();
632 m_jp2_image->y1 = image.
height();
634 ScanLineConverter scl(convFormat);
635 if (prec < 32 && isFloat) {
638 if (cs == OPJ_CLRSPC_SRGB) {
643 for (qint32 c = 0; c < ncomp; ++c) {
644 auto &&comp = m_jp2_image->comps[c];
645 auto mul = ncomp == 1 ? 1 : 4;
646 for (qint32 y = 0, h = image.
height(); y < h; ++y) {
648 auto ptr =
reinterpret_cast<const quint8 *
>(scl.convertedScanLine(image, y));
649 for (qint32 x = 0, w = image.
width(); x < w; ++x)
650 comp.data[y * w + x] = ptr[x * mul + c];
651 }
else if (prec == 16) {
652 auto ptr =
reinterpret_cast<const quint16 *
>(scl.convertedScanLine(image, y));
653 for (qint32 x = 0, w = image.
width(); x < w; ++x)
654 comp.data[y * w + x] = ptr[x * mul + c];
655 }
else if (prec == 32) {
656 auto ptr =
reinterpret_cast<const quint32 *
>(scl.convertedScanLine(image, y));
657 for (qint32 x = 0, w = image.
width(); x < w; ++x)
658 comp.data[y * w + x] = ptr[x * mul + c];
663 if (opjVersion() >= QT_VERSION_CHECK(2, 5, 4)) {
664 auto colorSpace = scl.targetColorSpace().iccProfile();
665 if (!colorSpace.isEmpty()) {
666 m_jp2_image->icc_profile_buf =
reinterpret_cast<OPJ_BYTE *
>(malloc(colorSpace.size()));
667 if (m_jp2_image->icc_profile_buf) {
668 memcpy(m_jp2_image->icc_profile_buf, colorSpace.data(), colorSpace.size());
669 m_jp2_image->icc_profile_len = colorSpace.size();
677 bool writeImage(QIODevice *device,
const QImage &image)
679 if (!imageToJp2(image)) {
680 qCritical() <<
"Error while creating JP2 image!";
684 std::unique_ptr<opj_codec_t, std::function<void(opj_codec_t *)>> codec(opj_create_compress(encoderFormat()), opj_destroy_codec);
685 if (codec ==
nullptr) {
686 qCritical() <<
"Error while creating encoder!";
689 enableThreads(codec.get());
694 opj_set_error_handler(m_jp2_codec, error_callback,
nullptr);
696 if (!opj_setup_encoder(codec.get(), &m_cparameters, m_jp2_image)) {
700 if (!createStream(device,
false)) {
704 if (!opj_start_compress(codec.get(), m_jp2_image, m_jp2_stream)) {
707 if (!opj_encode(codec.get(), m_jp2_stream)) {
710 if (!opj_end_compress(codec.get(), m_jp2_stream)) {
719 opj_stream_t *m_jp2_stream;
721 opj_image_t *m_jp2_image;
723 qint32 m_jp2_version;
726 opj_codec_t *m_jp2_codec;
728 opj_dparameters_t m_dparameters;
731 opj_cparameters_t m_cparameters;
735 QByteArray m_subtype;
739JP2Handler::JP2Handler()
741 , d(new JP2HandlerPrivate)
745bool JP2Handler::canRead()
const
747 if (canRead(device())) {
754bool JP2Handler::canRead(
QIODevice *device)
757 qWarning(
"JP2Handler::canRead() called with no device");
765 JP2HandlerPrivate handler;
766 return handler.detectDecoderFormat(device) != OPJ_CODEC_UNKNOWN;
769bool JP2Handler::read(
QImage *image)
772 if (dev ==
nullptr) {
775 auto img = d->readImage(dev);
783bool JP2Handler::write(
const QImage &image)
789 if (dev ==
nullptr) {
792 return d->writeImage(dev, image);
795bool JP2Handler::supportsOption(ImageOption option)
const
815void JP2Handler::setOption(ImageOption option,
const QVariant &value)
824 auto q = value.
toInt(&ok);
831QVariant JP2Handler::option(ImageOption option)
const
836 if (d->readHeader(device())) {
842 if (d->readHeader(device())) {
864 if (format ==
"jp2" || format ==
"j2k") {
868 if (format ==
"jpf") {
879 if (device->
isReadable() && JP2Handler::canRead(device)) {
896#include "moc_jp2_p.cpp"
Type type(const QSqlDatabase &db)
QFlags< Capability > Capabilities
QByteArray fromHex(const QByteArray &hexEncoded)
bool isEmpty() const const
QColorSpace fromIccProfile(const QByteArray &iccProfile)
bool isValid() const const
QColorSpace colorSpace() const const
bool hasAlphaChannel() const const
bool isGrayscale() const const
bool isNull() const const
void setColorSpace(const QColorSpace &colorSpace)
void setDevice(QIODevice *device)
bool isOpen() const const
bool isReadable() const const
virtual bool isSequential() const const
bool isWritable() const const
QByteArray peek(qint64 maxSize)
virtual qint64 size() const const
QString fromLatin1(QByteArrayView str)
QStringList split(QChar sep, Qt::SplitBehavior behavior, Qt::CaseSensitivity cs) const const
QVariant fromValue(T &&value)
QByteArray toByteArray() const const
int toInt(bool *ok) const const