From 5bbf16e87ca09369174c9d2bb7f7fe94b6d70275 Mon Sep 17 00:00:00 2001 From: Carl Hetherington Date: Sat, 26 Nov 2016 15:14:11 +0000 Subject: Reduce during J2K decode where possible for playback (#986). --- src/lib/j2k_image_proxy.cc | 49 +++++++++++++++++++++++++++------------------- 1 file changed, 29 insertions(+), 20 deletions(-) (limited to 'src/lib/j2k_image_proxy.cc') diff --git a/src/lib/j2k_image_proxy.cc b/src/lib/j2k_image_proxy.cc index fb38c9e99..4e38213c8 100644 --- a/src/lib/j2k_image_proxy.cc +++ b/src/lib/j2k_image_proxy.cc @@ -30,12 +30,14 @@ #include #include #include +#include #include #include "i18n.h" using std::string; using std::cout; +using std::max; using boost::shared_ptr; using boost::optional; using boost::dynamic_pointer_cast; @@ -91,46 +93,53 @@ J2KImageProxy::J2KImageProxy (shared_ptr xml, shared_ptr soc socket->read (_data.data().get (), _data.size ()); } -void -J2KImageProxy::ensure_j2k () const -{ - if (!_j2k) { - _j2k = dcp::decompress_j2k (const_cast (_data.data().get()), _data.size (), 0); - } -} - shared_ptr -J2KImageProxy::image (optional) const +J2KImageProxy::image (optional, optional target_size) const { - ensure_j2k (); - - if (_j2k->precision(0) < 12) { - int const shift = 12 - _j2k->precision (0); - for (int c = 0; c < 3; ++c) { - int* p = _j2k->data (c); - for (int y = 0; y < _j2k->size().height; ++y) { - for (int x = 0; x < _j2k->size().width; ++x) { - *p++ <<= shift; + if (!_j2k || target_size != _j2k_target_size) { + int reduce = 0; + + while (target_size && (_size.width / pow(2, reduce)) > target_size->width && (_size.height / pow(2, reduce)) > target_size->height) { + ++reduce; + } + + --reduce; + reduce = max (0, reduce); + _j2k = dcp::decompress_j2k (const_cast (_data.data().get()), _data.size (), reduce); + + if (_j2k->precision(0) < 12) { + int const shift = 12 - _j2k->precision (0); + for (int c = 0; c < 3; ++c) { + int* p = _j2k->data (c); + for (int y = 0; y < _j2k->size().height; ++y) { + for (int x = 0; x < _j2k->size().width; ++x) { + *p++ <<= shift; + } } } } + + _j2k_target_size = target_size; } - shared_ptr image (new Image (_pixel_format, _size, true)); + shared_ptr image (new Image (_pixel_format, _j2k->size(), true)); /* Copy data in whatever format (sRGB or XYZ) into our Image; I'm assuming the data is 12-bit either way. */ + int const width = _j2k->size().width; + int p = 0; for (int y = 0; y < _j2k->size().height; ++y) { uint16_t* q = (uint16_t *) (image->data()[0] + y * image->stride()[0]); - for (int x = 0; x < _j2k->size().width; ++x) { + for (int x = 0; x < width; ++x) { for (int c = 0; c < 3; ++c) { *q++ = _j2k->data(c)[p] << 4; } ++p; } + p += _j2k->factor(0) * width; } return image; -- cgit v1.2.3