aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorPierre Ossman <ossman@cendio.se>2020-08-19 13:49:34 +0200
committerPierre Ossman <ossman@cendio.se>2020-08-19 13:49:34 +0200
commit112f37f194cf1ce52ee10e2a6b4e71876c9e1c2d (patch)
treebdf8dd1594b8f63768aec402dabfeb59d9f3df36
parent82fbecb8a23bfdd3358516b1e6c98fe1df613791 (diff)
parentae38d967ea4aac937e6fce5c0aa00a4fc552b637 (diff)
downloadtigervnc-112f37f194cf1ce52ee10e2a6b4e71876c9e1c2d.tar.gz
tigervnc-112f37f194cf1ce52ee10e2a6b4e71876c9e1c2d.zip
Merge branch 'master-improve-ComparingUpdateTracker' of https://github.com/MarkMielke/tigervnc
-rw-r--r--common/rfb/ComparingUpdateTracker.cxx84
1 files changed, 80 insertions, 4 deletions
diff --git a/common/rfb/ComparingUpdateTracker.cxx b/common/rfb/ComparingUpdateTracker.cxx
index d5651200..14ed908c 100644
--- a/common/rfb/ComparingUpdateTracker.cxx
+++ b/common/rfb/ComparingUpdateTracker.cxx
@@ -120,6 +120,10 @@ void ComparingUpdateTracker::compareRect(const Rect& r, Region* newChanged)
rdr::U8* oldData = oldFb.getBufferRW(r, &oldStride);
int oldStrideBytes = oldStride * bytesPerPixel;
+ // Used to efficiently crop the left and right of the change rectangle
+ int minCompareWidthInPixels = BLOCK_SIZE / 8;
+ int minCompareWidthInBytes = minCompareWidthInPixels * bytesPerPixel;
+
for (int blockTop = r.tl.y; blockTop < r.br.y; blockTop += BLOCK_SIZE)
{
// Get a strip of the source buffer
@@ -139,19 +143,91 @@ void ComparingUpdateTracker::compareRect(const Rect& r, Region* newChanged)
int blockRight = __rfbmin(blockLeft+BLOCK_SIZE, r.br.x);
int blockWidthInBytes = (blockRight-blockLeft) * bytesPerPixel;
+ // Scan the block top to bottom, to identify the first row of change
for (int y = blockTop; y < blockBottom; y++)
{
if (memcmp(oldPtr, newPtr, blockWidthInBytes) != 0)
{
- // A block has changed - copy the remainder to the oldFb
- newChanged->assign_union(Region(Rect(blockLeft, blockTop,
- blockRight, blockBottom)));
- for (int y2 = y; y2 < blockBottom; y2++)
+ // Define the change rectangle using pessimistic values to start
+ int changeHeight = blockBottom - y;
+ int changeLeft = blockLeft;
+ int changeRight = blockRight;
+
+ // For every unchanged row at the bottom of the block, decrement change height
+ {
+ const rdr::U8* newRowPtr = newPtr + ((changeHeight - 1) * newStrideBytes);
+ const rdr::U8* oldRowPtr = oldPtr + ((changeHeight - 1) * oldStrideBytes);
+ while (changeHeight > 1 && memcmp(oldRowPtr, newRowPtr, blockWidthInBytes) == 0)
+ {
+ newRowPtr -= newStrideBytes;
+ oldRowPtr -= oldStrideBytes;
+
+ changeHeight--;
+ }
+ }
+
+ // For every unchanged column at the left of the block, increment change left
+ {
+ const rdr::U8* newColumnPtr = newPtr;
+ const rdr::U8* oldColumnPtr = oldPtr;
+ while (changeLeft + minCompareWidthInPixels < changeRight)
+ {
+ const rdr::U8* newRowPtr = newColumnPtr;
+ const rdr::U8* oldRowPtr = oldColumnPtr;
+ for (int row = 0; row < changeHeight; row++)
+ {
+ if (memcmp(oldRowPtr, newRowPtr, minCompareWidthInBytes) != 0)
+ goto endOfChangeLeft;
+
+ newRowPtr += newStrideBytes;
+ oldRowPtr += oldStrideBytes;
+ }
+
+ newColumnPtr += minCompareWidthInBytes;
+ oldColumnPtr += minCompareWidthInBytes;
+
+ changeLeft += minCompareWidthInPixels;
+ }
+ }
+ endOfChangeLeft:
+
+ // For every unchanged column at the right of the block, decrement change right
+ {
+ const rdr::U8* newColumnPtr = newPtr + blockWidthInBytes;
+ const rdr::U8* oldColumnPtr = oldPtr + blockWidthInBytes;
+ while (changeLeft + minCompareWidthInPixels < changeRight)
+ {
+ newColumnPtr -= minCompareWidthInBytes;
+ oldColumnPtr -= minCompareWidthInBytes;
+
+ const rdr::U8* newRowPtr = newColumnPtr;
+ const rdr::U8* oldRowPtr = oldColumnPtr;
+ for (int row = 0; row < changeHeight; row++)
+ {
+ if (memcmp(oldRowPtr, newRowPtr, minCompareWidthInBytes) != 0)
+ goto endOfChangeRight;
+
+ newRowPtr += newStrideBytes;
+ oldRowPtr += oldStrideBytes;
+ }
+
+ changeRight -= minCompareWidthInPixels;
+ }
+ }
+ endOfChangeRight:
+
+ // Block change extends from (changeLeft, y) to (changeRight, y + changeHeight)
+ newChanged->assign_union(Region(Rect(changeLeft, y, changeRight, y + changeHeight)));
+
+ // Copy the change from fb to oldFb to allow future changes to be identified
+ for (int row = 0; row < changeHeight; row++)
{
memcpy(oldPtr, newPtr, blockWidthInBytes);
newPtr += newStrideBytes;
oldPtr += oldStrideBytes;
}
+
+ // No further processing is required for this block
break;
}