diff options
-rw-r--r-- | rfb/TightDecoder.cxx | 7 | ||||
-rw-r--r-- | rfb/tightDecode.h | 72 |
2 files changed, 74 insertions, 5 deletions
diff --git a/rfb/TightDecoder.cxx b/rfb/TightDecoder.cxx index d634acb1..e51558fe 100644 --- a/rfb/TightDecoder.cxx +++ b/rfb/TightDecoder.cxx @@ -26,7 +26,12 @@ extern "C" { using namespace rfb; -#define RGB24_TO_PIXEL(bpp,r,g,b) \ +#define RGB_TO_PIXEL(r,g,b) \ + (((PIXEL_T)(r) & myFormat.redMax) << myFormat.redShift | \ + ((PIXEL_T)(g) & myFormat.greenMax) << myFormat.greenShift | \ + ((PIXEL_T)(b) & myFormat.blueMax) << myFormat.blueShift) + +#define RGB24_TO_PIXEL(r,g,b) \ ((((PIXEL_T)(r) & 0xFF) * myFormat.redMax + 127) / 255 \ << myFormat.redShift | \ (((PIXEL_T)(g) & 0xFF) * myFormat.greenMax + 127) / 255 \ diff --git a/rfb/tightDecode.h b/rfb/tightDecode.h index d066a558..a6b36f86 100644 --- a/rfb/tightDecode.h +++ b/rfb/tightDecode.h @@ -47,6 +47,8 @@ namespace rfb { #define TIGHT_MIN_TO_COMPRESS 12 static bool DecompressJpegRect(const Rect& r, rdr::InStream* is, PIXEL_T* buf, CMsgHandler* handler); +static void FilterGradient(const Rect& r, rdr::InStream* is, int dataSize, + PIXEL_T* buf, CMsgHandler* handler); // Main function implementing Tight decoder @@ -88,7 +90,7 @@ void TIGHT_DECODE (const Rect& r, rdr::InStream* is, // "Basic" compression type. int palSize = 0; - PIXEL_T palette[256]; + static PIXEL_T palette[256]; bool useGradient = false; if ((comp_ctl & rfbTightExplicitFilter) != 0) { @@ -133,9 +135,10 @@ void TIGHT_DECODE (const Rect& r, rdr::InStream* is, if (palSize == 0) { // Truecolor data - input->readBytes(buf, dataSize); if (useGradient) { - // FIXME: Implement the "gradient" filter. + FilterGradient(r, input, dataSize, buf, handler); + } else { + input->readBytes(buf, dataSize); } } else { int x, y, b, w; @@ -216,7 +219,7 @@ DecompressJpegRect(const Rect& r, rdr::InStream* is, pixelPtr = (PIXEL_T*)(scanline); for (int dx = 0; dx < r.width(); dx++) { *pixelPtr++ = - RGB24_TO_PIXEL(BPP, scanline[dx*3], scanline[dx*3+1], scanline[dx*3+2]); + RGB24_TO_PIXEL(scanline[dx*3], scanline[dx*3+1], scanline[dx*3+2]); } scanline += bytesPerRow; } @@ -234,6 +237,67 @@ DecompressJpegRect(const Rect& r, rdr::InStream* is, } +static void +FilterGradient(const Rect& r, rdr::InStream* is, int dataSize, + PIXEL_T* buf, CMsgHandler* handler) +{ + int x, y, c; + static PIXEL_T prevRow[TIGHT_MAX_WIDTH*sizeof(PIXEL_T)]; + static PIXEL_T thisRow[TIGHT_MAX_WIDTH*sizeof(PIXEL_T)]; + int pix[3]; + int max[3]; + int shift[3]; + int est[3]; + + memset(prevRow, 0, sizeof(prevRow)); + + // Allocate netbuf and read in data + PIXEL_T *netbuf = (PIXEL_T*)new rdr::U8[dataSize]; + if (!netbuf) + throw Exception("rfb::tightDecode unable to allocate buffer"); + is->readBytes(netbuf, dataSize); + + // Set up shortcut variables + const rfb::PixelFormat& myFormat = handler->cp.pf(); + max[0] = myFormat.redMax; + max[1] = myFormat.greenMax; + max[2] = myFormat.blueMax; + shift[0] = myFormat.redShift; + shift[1] = myFormat.greenShift; + shift[2] = myFormat.blueShift; + int rectHeight = r.height(); + int rectWidth = r.width(); + + for (y = 0; y < rectHeight; y++) { + /* First pixel in a row */ + for (c = 0; c < 3; c++) { + pix[c] = (netbuf[y*rectWidth] >> shift[c]) + prevRow[c] & max[c]; + thisRow[c] = pix[c]; + } + buf[y*rectWidth] = RGB_TO_PIXEL(pix[0], pix[1], pix[2]); + + /* Remaining pixels of a row */ + for (x = 1; x < rectWidth; x++) { + for (c = 0; c < 3; c++) { + est[c] = prevRow[x*3+c] + pix[c] - prevRow[(x-1)*3+c]; + if (est[c] > max[c]) { + est[c] = max[c]; + } else if (est[c] < 0) { + est[c] = 0; + } + pix[c] = (netbuf[y*rectWidth+x] >> shift[c]) + est[c] & max[c]; + thisRow[x*3+c] = pix[c]; + } + buf[y*rectWidth+x] = RGB_TO_PIXEL(pix[0], pix[1], pix[2]); + } + + memcpy(prevRow, thisRow, sizeof(prevRow)); + } + + delete [] netbuf; +} + +#undef TIGHT_MIN_TO_COMPRESS #undef TIGHT_DECODE #undef READ_PIXEL #undef PIXEL_T |