X-Git-Url: https://git.carlh.net/gitweb/?a=blobdiff_plain;f=src%2Flib%2Fj2k_image_proxy.cc;h=eea4b038a2c841861a5799288bb216cb7ee9cafc;hb=c180f317d5a8b27dd191c1f2228ceb6fc4039393;hp=fb38c9e999a9d04913f14823a67a0ed581429944;hpb=b81241ce69a689629307832f802ac4faa6ed885f;p=dcpomatic.git diff --git a/src/lib/j2k_image_proxy.cc b/src/lib/j2k_image_proxy.cc index fb38c9e99..eea4b038a 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,41 +93,47 @@ 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; }