diff options
author | Pierre Ossman <ossman@cendio.se> | 2011-10-18 14:27:07 +0000 |
---|---|---|
committer | Pierre Ossman <ossman@cendio.se> | 2011-10-18 14:27:07 +0000 |
commit | b5822f32ab5043b4e70e55ccbac0f548a2f784fc (patch) | |
tree | 9746e77ebc397ca95a31896e604e3de91c49fa73 | |
parent | 308b7eda345fc8b8a4357518cdd3612762c35aa2 (diff) | |
download | tigervnc-b5822f32ab5043b4e70e55ccbac0f548a2f784fc.tar.gz tigervnc-b5822f32ab5043b4e70e55ccbac0f548a2f784fc.zip |
Another attempt at solving the compression level change problems. We were still
not detecting the correct approach properly, and hence still getting crashes.
git-svn-id: svn://svn.code.sf.net/p/tigervnc/code/trunk@4731 3789f03b-4d11-0410-bbf8-ca57d06f2519
-rw-r--r-- | common/rdr/ZlibOutStream.cxx | 136 | ||||
-rw-r--r-- | common/rdr/ZlibOutStream.h | 2 |
2 files changed, 75 insertions, 63 deletions
diff --git a/common/rdr/ZlibOutStream.cxx b/common/rdr/ZlibOutStream.cxx index f04536fa..260bd1a4 100644 --- a/common/rdr/ZlibOutStream.cxx +++ b/common/rdr/ZlibOutStream.cxx @@ -23,27 +23,28 @@ #include <zlib.h> +#undef ZLIBOUT_DEBUG + using namespace rdr; enum { DEFAULT_BUF_SIZE = 16384 }; ZlibOutStream::ZlibOutStream(OutStream* os, int bufSize_, int compressLevel) : underlying(os), compressionLevel(compressLevel), newLevel(compressLevel), - bufSize(bufSize_ ? bufSize_ : DEFAULT_BUF_SIZE), offset(0), - newBehavior(false) + bufSize(bufSize_ ? bufSize_ : DEFAULT_BUF_SIZE), offset(0) { zs = new z_stream; zs->zalloc = Z_NULL; zs->zfree = Z_NULL; zs->opaque = Z_NULL; + zs->next_in = Z_NULL; + zs->avail_in = 0; if (deflateInit(zs, compressLevel) != Z_OK) { delete zs; throw Exception("ZlibOutStream: deflateInit failed"); } ptr = start = new U8[bufSize]; end = start + bufSize; - const char *version = zlibVersion(); - if (strcmp(version, "1.2.3") > 0) newBehavior = true; } ZlibOutStream::~ZlibOutStream() @@ -77,35 +78,17 @@ int ZlibOutStream::length() void ZlibOutStream::flush() { + checkCompressionLevel(); + zs->next_in = start; zs->avail_in = ptr - start; -// fprintf(stderr,"zos flush: avail_in %d\n",zs->avail_in); - - if (!underlying) - throw Exception("ZlibOutStream: underlying OutStream has not been set"); - - while (zs->avail_in != 0) { - - do { - underlying->check(1); - zs->next_out = underlying->getptr(); - zs->avail_out = underlying->getend() - underlying->getptr(); - -// fprintf(stderr,"zos flush: calling deflate, avail_in %d, avail_out %d\n", -// zs->avail_in,zs->avail_out); - checkCompressionLevel(); - if (zs->avail_in != 0) { - int rc = deflate(zs, Z_SYNC_FLUSH); - if (rc != Z_OK) throw Exception("ZlibOutStream: deflate failed"); - } +#ifdef ZLIBOUT_DEBUG + fprintf(stderr,"zos flush: avail_in %d\n",zs->avail_in); +#endif -// fprintf(stderr,"zos flush: after deflate: %d bytes\n", -// zs->next_out-underlying->getptr()); - - underlying->setptr(zs->next_out); - } while (zs->avail_out == 0); - } + // Force out everything from the zlib encoder + deflate(Z_SYNC_FLUSH); offset += ptr - start; ptr = start; @@ -113,37 +96,20 @@ void ZlibOutStream::flush() int ZlibOutStream::overrun(int itemSize, int nItems) { -// fprintf(stderr,"ZlibOutStream overrun\n"); +#ifdef ZLIBOUT_DEBUG + fprintf(stderr,"zos overrun\n"); +#endif if (itemSize > bufSize) throw Exception("ZlibOutStream overrun: max itemSize exceeded"); - if (!underlying) - throw Exception("ZlibOutStream: underlying OutStream has not been set"); + checkCompressionLevel(); while (end - ptr < itemSize) { zs->next_in = start; zs->avail_in = ptr - start; - do { - underlying->check(1); - zs->next_out = underlying->getptr(); - zs->avail_out = underlying->getend() - underlying->getptr(); - -// fprintf(stderr,"zos overrun: calling deflate, avail_in %d, avail_out %d\n", -// zs->avail_in,zs->avail_out); - - checkCompressionLevel(); - if (zs->avail_in != 0) { - int rc = deflate(zs, 0); - if (rc != Z_OK) throw Exception("ZlibOutStream: deflate failed"); - } - -// fprintf(stderr,"zos overrun: after deflate: %d bytes\n", -// zs->next_out-underlying->getptr()); - - underlying->setptr(zs->next_out); - } while (zs->avail_out == 0); + deflate(Z_NO_FLUSH); // output buffer not full @@ -166,23 +132,69 @@ int ZlibOutStream::overrun(int itemSize, int nItems) return nItems; } -void ZlibOutStream::checkCompressionLevel() +void ZlibOutStream::deflate(int flush) { - if (newLevel != compressionLevel) { + int rc; + + if (!underlying) + throw Exception("ZlibOutStream: underlying OutStream has not been set"); + + if ((flush == Z_NO_FLUSH) && (zs->avail_in == 0)) + return; + + do { + underlying->check(1); + zs->next_out = underlying->getptr(); + zs->avail_out = underlying->getend() - underlying->getptr(); + +#ifdef ZLIBOUT_DEBUG + fprintf(stderr,"zos: calling deflate, avail_in %d, avail_out %d\n", + zs->avail_in,zs->avail_out); +#endif - // This is a horrible hack, but after many hours of trying, I couldn't find - // a better way to make this class work properly with both Zlib 1.2.3 and - // 1.2.5. 1.2.3 does a Z_PARTIAL_FLUSH in the body of deflateParams() if - // the compression level has changed, and 1.2.5 does a Z_BLOCK flush. + rc = ::deflate(zs, flush); + if (rc != Z_OK) { + // Silly zlib returns an error if you try to flush something twice + if ((rc == Z_BUF_ERROR) && (flush != Z_NO_FLUSH)) + break; - if (newBehavior) { - int rc = deflate(zs, Z_SYNC_FLUSH); - if (rc != Z_OK) throw Exception("ZlibOutStream: deflate failed"); + throw Exception("ZlibOutStream: deflate failed"); } - if (deflateParams (zs, newLevel, Z_DEFAULT_STRATEGY) != Z_OK) { - throw Exception("ZlibOutStream: deflateParams failed"); +#ifdef ZLIBOUT_DEBUG + fprintf(stderr,"zos: after deflate: %d bytes\n", + zs->next_out-underlying->getptr()); +#endif + + underlying->setptr(zs->next_out); + } while (zs->avail_out == 0); +} + +void ZlibOutStream::checkCompressionLevel() +{ + int rc; + + if (newLevel != compressionLevel) { +#ifdef ZLIBOUT_DEBUG + fprintf(stderr,"zos change: avail_in %d\n",zs->avail_in); +#endif + + // zlib is just horribly stupid. It does an implicit flush on + // parameter changes, but the flush it does is not one that forces + // out all the data. And since you cannot flush things again, we + // cannot force out our data after the parameter change. Hence we + // need to do a more proper flush here first. + deflate(Z_SYNC_FLUSH); + + rc = deflateParams (zs, newLevel, Z_DEFAULT_STRATEGY); + if (rc != Z_OK) { + // The implicit flush can result in this error, caused by the + // explicit flush we did above. It should be safe to ignore though + // as the first flush should have left things in a stable state... + if (rc != Z_BUF_ERROR) + throw Exception("ZlibOutStream: deflateParams failed"); } + compressionLevel = newLevel; } } diff --git a/common/rdr/ZlibOutStream.h b/common/rdr/ZlibOutStream.h index 250bb035..2d82a13b 100644 --- a/common/rdr/ZlibOutStream.h +++ b/common/rdr/ZlibOutStream.h @@ -46,6 +46,7 @@ namespace rdr { private: int overrun(int itemSize, int nItems); + void deflate(int flush); void checkCompressionLevel(); OutStream* underlying; @@ -55,7 +56,6 @@ namespace rdr { int offset; z_stream_s* zs; U8* start; - bool newBehavior; }; } // end of namespace rdr |