Blame libfreerdp/codec/progressive.c

Packit 1fb8d4
/**
Packit 1fb8d4
 * FreeRDP: A Remote Desktop Protocol Implementation
Packit 1fb8d4
 * Progressive Codec Bitmap Compression
Packit 1fb8d4
 *
Packit 1fb8d4
 * Copyright 2014 Marc-Andre Moreau <marcandre.moreau@gmail.com>
Packit 1fb8d4
 *
Packit 1fb8d4
 * Licensed under the Apache License, Version 2.0 (the "License");
Packit 1fb8d4
 * you may not use this file except in compliance with the License.
Packit 1fb8d4
 * You may obtain a copy of the License at
Packit 1fb8d4
 *
Packit 1fb8d4
 *     http://www.apache.org/licenses/LICENSE-2.0
Packit 1fb8d4
 *
Packit 1fb8d4
 * Unless required by applicable law or agreed to in writing, software
Packit 1fb8d4
 * distributed under the License is distributed on an "AS IS" BASIS,
Packit 1fb8d4
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Packit 1fb8d4
 * See the License for the specific language governing permissions and
Packit 1fb8d4
 * limitations under the License.
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
#ifdef HAVE_CONFIG_H
Packit 1fb8d4
#include "config.h"
Packit 1fb8d4
#endif
Packit 1fb8d4
Packit 1fb8d4
#include <winpr/crt.h>
Packit 1fb8d4
#include <winpr/print.h>
Packit 1fb8d4
#include <winpr/bitstream.h>
Packit 1fb8d4
Packit 1fb8d4
#include <freerdp/primitives.h>
Packit 1fb8d4
#include <freerdp/codec/color.h>
Packit 1fb8d4
#include <freerdp/codec/progressive.h>
Packit 1fb8d4
#include <freerdp/codec/region.h>
Packit 1fb8d4
#include <freerdp/log.h>
Packit 1fb8d4
Packit 1fb8d4
#include "rfx_differential.h"
Packit 1fb8d4
#include "rfx_quantization.h"
Packit 1fb8d4
#include "rfx_rlgr.h"
Packit 1fb8d4
#include "progressive.h"
Packit 1fb8d4
Packit 1fb8d4
#define TAG FREERDP_TAG("codec.progressive")
Packit 1fb8d4
Packit 1fb8d4
struct _RFX_PROGRESSIVE_UPGRADE_STATE
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL nonLL;
Packit 1fb8d4
	wBitStream* srl;
Packit 1fb8d4
	wBitStream* raw;
Packit 1fb8d4
Packit 1fb8d4
	/* SRL state */
Packit 1fb8d4
Packit 1fb8d4
	int kp;
Packit 1fb8d4
	int nz;
Packit 1fb8d4
	BOOL mode;
Packit 1fb8d4
};
Packit 1fb8d4
typedef struct _RFX_PROGRESSIVE_UPGRADE_STATE RFX_PROGRESSIVE_UPGRADE_STATE;
Packit 1fb8d4
Packit 1fb8d4
static const char* progressive_get_block_type_string(UINT16 blockType)
Packit 1fb8d4
{
Packit 1fb8d4
	switch (blockType)
Packit 1fb8d4
	{
Packit 1fb8d4
		case PROGRESSIVE_WBT_SYNC:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_SYNC";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_FRAME_BEGIN:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_FRAME_BEGIN";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_FRAME_END:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_FRAME_END";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_CONTEXT:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_CONTEXT";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_REGION:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_REGION";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_TILE_SIMPLE:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_TILE_SIMPLE";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_TILE_FIRST:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_TILE_FIRST";
Packit 1fb8d4
Packit 1fb8d4
		case PROGRESSIVE_WBT_TILE_UPGRADE:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_TILE_UPGRADE";
Packit 1fb8d4
Packit 1fb8d4
		default:
Packit 1fb8d4
			return "PROGRESSIVE_WBT_UNKNOWN";
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return "PROGRESSIVE_WBT_UNKNOWN";
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_component_codec_quant_read(const BYTE* block,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* quantVal)
Packit 1fb8d4
{
Packit 1fb8d4
	quantVal->LL3 = block[0] & 0x0F;
Packit 1fb8d4
	quantVal->HL3 = block[0] >> 4;
Packit 1fb8d4
	quantVal->LH3 = block[1] & 0x0F;
Packit 1fb8d4
	quantVal->HH3 = block[1] >> 4;
Packit 1fb8d4
	quantVal->HL2 = block[2] & 0x0F;
Packit 1fb8d4
	quantVal->LH2 = block[2] >> 4;
Packit 1fb8d4
	quantVal->HH2 = block[3] & 0x0F;
Packit 1fb8d4
	quantVal->HL1 = block[3] >> 4;
Packit 1fb8d4
	quantVal->LH1 = block[4] & 0x0F;
Packit 1fb8d4
	quantVal->HH1 = block[4] >> 4;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_quant_ladd(RFX_COMPONENT_CODEC_QUANT* q, int val)
Packit 1fb8d4
{
Packit 1fb8d4
	q->HL1 += val; /* HL1 */
Packit 1fb8d4
	q->LH1 += val; /* LH1 */
Packit 1fb8d4
	q->HH1 += val; /* HH1 */
Packit 1fb8d4
	q->HL2 += val; /* HL2 */
Packit 1fb8d4
	q->LH2 += val; /* LH2 */
Packit 1fb8d4
	q->HH2 += val; /* HH2 */
Packit 1fb8d4
	q->HL3 += val; /* HL3 */
Packit 1fb8d4
	q->LH3 += val; /* LH3 */
Packit 1fb8d4
	q->HH3 += val; /* HH3 */
Packit 1fb8d4
	q->LL3 += val; /* LL3 */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_quant_add(RFX_COMPONENT_CODEC_QUANT* q1,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* q2,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* dst)
Packit 1fb8d4
{
Packit 1fb8d4
	dst->HL1 = q1->HL1 + q2->HL1; /* HL1 */
Packit 1fb8d4
	dst->LH1 = q1->LH1 + q2->LH1; /* LH1 */
Packit 1fb8d4
	dst->HH1 = q1->HH1 + q2->HH1; /* HH1 */
Packit 1fb8d4
	dst->HL2 = q1->HL2 + q2->HL2; /* HL2 */
Packit 1fb8d4
	dst->LH2 = q1->LH2 + q2->LH2; /* LH2 */
Packit 1fb8d4
	dst->HH2 = q1->HH2 + q2->HH2; /* HH2 */
Packit 1fb8d4
	dst->HL3 = q1->HL3 + q2->HL3; /* HL3 */
Packit 1fb8d4
	dst->LH3 = q1->LH3 + q2->LH3; /* LH3 */
Packit 1fb8d4
	dst->HH3 = q1->HH3 + q2->HH3; /* HH3 */
Packit 1fb8d4
	dst->LL3 = q1->LL3 + q2->LL3; /* LL3 */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_quant_lsub(RFX_COMPONENT_CODEC_QUANT* q, int val)
Packit 1fb8d4
{
Packit 1fb8d4
	q->HL1 -= val; /* HL1 */
Packit 1fb8d4
	q->LH1 -= val; /* LH1 */
Packit 1fb8d4
	q->HH1 -= val; /* HH1 */
Packit 1fb8d4
	q->HL2 -= val; /* HL2 */
Packit 1fb8d4
	q->LH2 -= val; /* LH2 */
Packit 1fb8d4
	q->HH2 -= val; /* HH2 */
Packit 1fb8d4
	q->HL3 -= val; /* HL3 */
Packit 1fb8d4
	q->LH3 -= val; /* LH3 */
Packit 1fb8d4
	q->HH3 -= val; /* HH3 */
Packit 1fb8d4
	q->LL3 -= val; /* LL3 */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_quant_sub(RFX_COMPONENT_CODEC_QUANT* q1,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* q2,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* dst)
Packit 1fb8d4
{
Packit 1fb8d4
	dst->HL1 = q1->HL1 - q2->HL1; /* HL1 */
Packit 1fb8d4
	dst->LH1 = q1->LH1 - q2->LH1; /* LH1 */
Packit 1fb8d4
	dst->HH1 = q1->HH1 - q2->HH1; /* HH1 */
Packit 1fb8d4
	dst->HL2 = q1->HL2 - q2->HL2; /* HL2 */
Packit 1fb8d4
	dst->LH2 = q1->LH2 - q2->LH2; /* LH2 */
Packit 1fb8d4
	dst->HH2 = q1->HH2 - q2->HH2; /* HH2 */
Packit 1fb8d4
	dst->HL3 = q1->HL3 - q2->HL3; /* HL3 */
Packit 1fb8d4
	dst->LH3 = q1->LH3 - q2->LH3; /* LH3 */
Packit 1fb8d4
	dst->HH3 = q1->HH3 - q2->HH3; /* HH3 */
Packit 1fb8d4
	dst->LL3 = q1->LL3 - q2->LL3; /* LL3 */
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_rfx_quant_lcmp_less_equal(RFX_COMPONENT_CODEC_QUANT* q,
Packit 1fb8d4
        int val)
Packit 1fb8d4
{
Packit 1fb8d4
	if (q->HL1 > val) return FALSE; /* HL1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH1 > val) return FALSE; /* LH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH1 > val) return FALSE; /* HH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HL2 > val) return FALSE; /* HL2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH2 > val) return FALSE; /* LH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH2 > val) return FALSE; /* HH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HL3 > val) return FALSE; /* HL3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH3 > val) return FALSE; /* LH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH3 > val) return FALSE; /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LL3 > val) return FALSE; /* LL3 */
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_rfx_quant_cmp_less_equal(RFX_COMPONENT_CODEC_QUANT* q1,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* q2)
Packit 1fb8d4
{
Packit 1fb8d4
	if (q1->HL1 > q2->HL1) return FALSE; /* HL1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH1 > q2->LH1) return FALSE; /* LH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH1 > q2->HH1) return FALSE; /* HH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL2 > q2->HL2) return FALSE; /* HL2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH2 > q2->LH2) return FALSE; /* LH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH2 > q2->HH2) return FALSE; /* HH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL3 > q2->HL3) return FALSE; /* HL3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH3 > q2->LH3) return FALSE; /* LH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH3 > q2->HH3) return FALSE; /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LL3 > q2->LL3) return FALSE; /* LL3 */
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_rfx_quant_lcmp_greater_equal(RFX_COMPONENT_CODEC_QUANT* q,
Packit 1fb8d4
        int val)
Packit 1fb8d4
{
Packit 1fb8d4
	if (q->HL1 < val) return FALSE; /* HL1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH1 < val) return FALSE; /* LH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH1 < val) return FALSE; /* HH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HL2 < val) return FALSE; /* HL2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH2 < val) return FALSE; /* LH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH2 < val) return FALSE; /* HH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HL3 < val) return FALSE; /* HL3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LH3 < val) return FALSE; /* LH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->HH3 < val) return FALSE; /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q->LL3 < val) return FALSE; /* LL3 */
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_rfx_quant_cmp_greater_equal(RFX_COMPONENT_CODEC_QUANT* q1,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* q2)
Packit 1fb8d4
{
Packit 1fb8d4
	if (q1->HL1 < q2->HL1) return FALSE; /* HL1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH1 < q2->LH1) return FALSE; /* LH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH1 < q2->HH1) return FALSE; /* HH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL2 < q2->HL2) return FALSE; /* HL2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH2 < q2->LH2) return FALSE; /* LH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH2 < q2->HH2) return FALSE; /* HH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL3 < q2->HL3) return FALSE; /* HL3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH3 < q2->LH3) return FALSE; /* LH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH3 < q2->HH3) return FALSE; /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LL3 < q2->LL3) return FALSE; /* LL3 */
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_rfx_quant_cmp_equal(RFX_COMPONENT_CODEC_QUANT* q1,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* q2)
Packit 1fb8d4
{
Packit 1fb8d4
	if (q1->HL1 != q2->HL1) return FALSE; /* HL1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH1 != q2->LH1) return FALSE; /* LH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH1 != q2->HH1) return FALSE; /* HH1 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL2 != q2->HL2) return FALSE; /* HL2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH2 != q2->LH2) return FALSE; /* LH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH2 != q2->HH2) return FALSE; /* HH2 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HL3 != q2->HL3) return FALSE; /* HL3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LH3 != q2->LH3) return FALSE; /* LH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->HH3 != q2->HH3) return FALSE; /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	if (q1->LL3 != q2->LL3) return FALSE; /* LL3 */
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void progressive_rfx_quant_print(RFX_COMPONENT_CODEC_QUANT* q, const char* name)
Packit 1fb8d4
{
Packit 1fb8d4
	fprintf(stderr,
Packit 1fb8d4
	        "%s: HL1: %"PRIu8" LH1: %"PRIu8" HH1: %"PRIu8" HL2: %"PRIu8" LH2: %"PRIu8" HH2: %"PRIu8" HL3: %"PRIu8" LH3: %"PRIu8" HH3: %"PRIu8" LL3: %"PRIu8"\n",
Packit 1fb8d4
	        name, q->HL1, q->LH1, q->HH1, q->HL2, q->LH2, q->HH2, q->HL3, q->LH3, q->HH3,
Packit 1fb8d4
	        q->LL3);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
Packit 1fb8d4
static INLINE BOOL progressive_set_surface_data(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        UINT16 surfaceId, void* pData)
Packit 1fb8d4
{
Packit 1fb8d4
	ULONG_PTR key;
Packit 1fb8d4
	key = ((ULONG_PTR) surfaceId) + 1;
Packit 1fb8d4
Packit 1fb8d4
	if (pData)
Packit 1fb8d4
		return HashTable_Add(progressive->SurfaceContexts, (void *)key, pData) >= 0;
Packit 1fb8d4
Packit 1fb8d4
	HashTable_Remove(progressive->SurfaceContexts, (void*) key);
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void* progressive_get_surface_data(PROGRESSIVE_CONTEXT* progressive, UINT16 surfaceId)
Packit 1fb8d4
{
Packit 1fb8d4
	ULONG_PTR key;
Packit 1fb8d4
	void* pData = NULL;
Packit 1fb8d4
	key = ((ULONG_PTR) surfaceId) + 1;
Packit 1fb8d4
	pData = HashTable_GetItemValue(progressive->SurfaceContexts, (void*) key);
Packit 1fb8d4
	return pData;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static PROGRESSIVE_SURFACE_CONTEXT* progressive_surface_context_new(UINT16 surfaceId, UINT32 width, UINT32 height)
Packit 1fb8d4
{
Packit 1fb8d4
	PROGRESSIVE_SURFACE_CONTEXT* surface;
Packit 1fb8d4
	surface = (PROGRESSIVE_SURFACE_CONTEXT*) calloc(
Packit 1fb8d4
	              1, sizeof(PROGRESSIVE_SURFACE_CONTEXT));
Packit 1fb8d4
Packit 1fb8d4
	if (!surface)
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
Packit 1fb8d4
	surface->id = surfaceId;
Packit 1fb8d4
	surface->width = width;
Packit 1fb8d4
	surface->height = height;
Packit 1fb8d4
	surface->gridWidth = (width + (64 - width % 64)) / 64;
Packit 1fb8d4
	surface->gridHeight = (height + (64 - height % 64)) / 64;
Packit 1fb8d4
	surface->gridSize = surface->gridWidth * surface->gridHeight;
Packit 1fb8d4
	surface->tiles = (RFX_PROGRESSIVE_TILE*) calloc(surface->gridSize, sizeof(RFX_PROGRESSIVE_TILE));
Packit 1fb8d4
Packit 1fb8d4
	if (!surface->tiles)
Packit 1fb8d4
	{
Packit 1fb8d4
		free(surface);
Packit 1fb8d4
		return NULL;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return surface;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static void progressive_surface_context_free(PROGRESSIVE_SURFACE_CONTEXT* surface)
Packit 1fb8d4
{
Packit 1fb8d4
	UINT32 index;
Packit 1fb8d4
	RFX_PROGRESSIVE_TILE* tile;
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < surface->gridSize; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		tile = &(surface->tiles[index]);
Packit 1fb8d4
Packit 1fb8d4
		if (tile->data)
Packit 1fb8d4
			_aligned_free(tile->data);
Packit 1fb8d4
Packit 1fb8d4
		if (tile->sign)
Packit 1fb8d4
			_aligned_free(tile->sign);
Packit 1fb8d4
Packit 1fb8d4
		if (tile->current)
Packit 1fb8d4
			_aligned_free(tile->current);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	free(surface->tiles);
Packit 1fb8d4
	free(surface);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
INT32 progressive_create_surface_context(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        UINT16 surfaceId, UINT32 width, UINT32 height)
Packit 1fb8d4
{
Packit 1fb8d4
	PROGRESSIVE_SURFACE_CONTEXT* surface;
Packit 1fb8d4
	surface = (PROGRESSIVE_SURFACE_CONTEXT *)progressive_get_surface_data(progressive, surfaceId);
Packit 1fb8d4
Packit 1fb8d4
	if (!surface)
Packit 1fb8d4
	{
Packit 1fb8d4
		surface = progressive_surface_context_new(surfaceId, width, height);
Packit 1fb8d4
Packit 1fb8d4
		if (!surface)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
Packit 1fb8d4
		if (!progressive_set_surface_data(progressive, surfaceId, (void*) surface))
Packit 1fb8d4
		{
Packit 1fb8d4
			progressive_surface_context_free(surface);
Packit 1fb8d4
			return -1;
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int progressive_delete_surface_context(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
                                       UINT16 surfaceId)
Packit 1fb8d4
{
Packit 1fb8d4
	PROGRESSIVE_SURFACE_CONTEXT* surface;
Packit 1fb8d4
	surface = (PROGRESSIVE_SURFACE_CONTEXT *)progressive_get_surface_data(progressive, surfaceId);
Packit 1fb8d4
Packit 1fb8d4
	if (surface)
Packit 1fb8d4
	{
Packit 1fb8d4
		progressive_set_surface_data(progressive, surfaceId, NULL);
Packit 1fb8d4
		progressive_surface_context_free(surface);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
/*
Packit 1fb8d4
 * Band		Offset		Dimensions	Size
Packit 1fb8d4
 *
Packit 1fb8d4
 * HL1		0		31x33		1023
Packit 1fb8d4
 * LH1		1023		33x31		1023
Packit 1fb8d4
 * HH1		2046		31x31		961
Packit 1fb8d4
 *
Packit 1fb8d4
 * HL2		3007		16x17		272
Packit 1fb8d4
 * LH2		3279		17x16		272
Packit 1fb8d4
 * HH2		3551		16x16		256
Packit 1fb8d4
 *
Packit 1fb8d4
 * HL3		3807		8x9		72
Packit 1fb8d4
 * LH3		3879		9x8		72
Packit 1fb8d4
 * HH3		3951		8x8		64
Packit 1fb8d4
 *
Packit 1fb8d4
 * LL3		4015		9x9		81
Packit 1fb8d4
 */
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_idwt_x(INT16* pLowBand, int nLowStep,
Packit 1fb8d4
        INT16* pHighBand,
Packit 1fb8d4
        int nHighStep, INT16* pDstBand, int nDstStep,
Packit 1fb8d4
        int nLowCount, int nHighCount, int nDstCount)
Packit 1fb8d4
{
Packit 1fb8d4
	int i, j;
Packit 1fb8d4
	INT16 L0;
Packit 1fb8d4
	INT16 H0, H1;
Packit 1fb8d4
	INT16 X0, X1, X2;
Packit 1fb8d4
	INT16* pL, *pH, *pX;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < nDstCount; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pL = pLowBand;
Packit 1fb8d4
		pH = pHighBand;
Packit 1fb8d4
		pX = pDstBand;
Packit 1fb8d4
		H0 = *pH;
Packit 1fb8d4
		pH++;
Packit 1fb8d4
		L0 = *pL;
Packit 1fb8d4
		pL++;
Packit 1fb8d4
		X0 = L0 - H0;
Packit 1fb8d4
		X2 = L0 - H0;
Packit 1fb8d4
Packit 1fb8d4
		for (j = 0; j < (nHighCount - 1); j++)
Packit 1fb8d4
		{
Packit 1fb8d4
			H1 = *pH;
Packit 1fb8d4
			pH++;
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			pL++;
Packit 1fb8d4
			X2 = L0 - ((H0 + H1) / 2);
Packit 1fb8d4
			X1 = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
			pX[0] = X0;
Packit 1fb8d4
			pX[1] = X1;
Packit 1fb8d4
			pX += 2;
Packit 1fb8d4
			X0 = X2;
Packit 1fb8d4
			H0 = H1;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (nLowCount <= (nHighCount + 1))
Packit 1fb8d4
		{
Packit 1fb8d4
			if (nLowCount <= nHighCount)
Packit 1fb8d4
			{
Packit 1fb8d4
				pX[0] = X2;
Packit 1fb8d4
				pX[1] = X2 + (2 * H0);
Packit 1fb8d4
			}
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				L0 = *pL;
Packit 1fb8d4
				pL++;
Packit 1fb8d4
				X0 = L0 - H0;
Packit 1fb8d4
				pX[0] = X2;
Packit 1fb8d4
				pX[1] = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
				pX[2] = X0;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			pL++;
Packit 1fb8d4
			X0 = L0 - (H0 / 2);
Packit 1fb8d4
			pX[0] = X2;
Packit 1fb8d4
			pX[1] = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
			pX[2] = X0;
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			pL++;
Packit 1fb8d4
			pX[3] = (X0 + L0) / 2;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pLowBand += nLowStep;
Packit 1fb8d4
		pHighBand += nHighStep;
Packit 1fb8d4
		pDstBand += nDstStep;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_idwt_y(INT16* pLowBand, int nLowStep,
Packit 1fb8d4
        INT16* pHighBand,
Packit 1fb8d4
        int nHighStep, INT16* pDstBand, int nDstStep,
Packit 1fb8d4
        int nLowCount, int nHighCount, int nDstCount)
Packit 1fb8d4
{
Packit 1fb8d4
	int i, j;
Packit 1fb8d4
	INT16 L0;
Packit 1fb8d4
	INT16 H0, H1;
Packit 1fb8d4
	INT16 X0, X1, X2;
Packit 1fb8d4
	INT16* pL, *pH, *pX;
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < nDstCount; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		pL = pLowBand;
Packit 1fb8d4
		pH = pHighBand;
Packit 1fb8d4
		pX = pDstBand;
Packit 1fb8d4
		H0 = *pH;
Packit 1fb8d4
		pH += nHighStep;
Packit 1fb8d4
		L0 = *pL;
Packit 1fb8d4
		pL += nLowStep;
Packit 1fb8d4
		X0 = L0 - H0;
Packit 1fb8d4
		X2 = L0 - H0;
Packit 1fb8d4
Packit 1fb8d4
		for (j = 0; j < (nHighCount - 1); j++)
Packit 1fb8d4
		{
Packit 1fb8d4
			H1 = *pH;
Packit 1fb8d4
			pH += nHighStep;
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			pL += nLowStep;
Packit 1fb8d4
			X2 = L0 - ((H0 + H1) / 2);
Packit 1fb8d4
			X1 = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
			*pX = X0;
Packit 1fb8d4
			pX += nDstStep;
Packit 1fb8d4
			*pX = X1;
Packit 1fb8d4
			pX += nDstStep;
Packit 1fb8d4
			X0 = X2;
Packit 1fb8d4
			H0 = H1;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (nLowCount <= (nHighCount + 1))
Packit 1fb8d4
		{
Packit 1fb8d4
			if (nLowCount <= nHighCount)
Packit 1fb8d4
			{
Packit 1fb8d4
				*pX = X2;
Packit 1fb8d4
				pX += nDstStep;
Packit 1fb8d4
				*pX = X2 + (2 * H0);
Packit 1fb8d4
			}
Packit 1fb8d4
			else
Packit 1fb8d4
			{
Packit 1fb8d4
				L0 = *pL;
Packit 1fb8d4
				X0 = L0 - H0;
Packit 1fb8d4
				*pX = X2;
Packit 1fb8d4
				pX += nDstStep;
Packit 1fb8d4
				*pX = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
				pX += nDstStep;
Packit 1fb8d4
				*pX = X0;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			pL += nLowStep;
Packit 1fb8d4
			X0 = L0 - (H0 / 2);
Packit 1fb8d4
			*pX = X2;
Packit 1fb8d4
			pX += nDstStep;
Packit 1fb8d4
			*pX = ((X0 + X2) / 2) + (2 * H0);
Packit 1fb8d4
			pX += nDstStep;
Packit 1fb8d4
			*pX = X0;
Packit 1fb8d4
			pX += nDstStep;
Packit 1fb8d4
			L0 = *pL;
Packit 1fb8d4
			*pX = (X0 + L0) / 2;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		pLowBand++;
Packit 1fb8d4
		pHighBand++;
Packit 1fb8d4
		pDstBand++;
Packit 1fb8d4
	}
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_get_band_l_count(int level)
Packit 1fb8d4
{
Packit 1fb8d4
	return (64 >> level) + 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_get_band_h_count(int level)
Packit 1fb8d4
{
Packit 1fb8d4
	if (level == 1)
Packit 1fb8d4
		return (64 >> 1) - 1;
Packit 1fb8d4
	else
Packit 1fb8d4
		return (64 + (1 << (level - 1))) >> level;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_dwt_2d_decode_block(INT16* buffer, INT16* temp,
Packit 1fb8d4
        int level)
Packit 1fb8d4
{
Packit 1fb8d4
	int offset;
Packit 1fb8d4
	int nBandL;
Packit 1fb8d4
	int nBandH;
Packit 1fb8d4
	int nDstStepX;
Packit 1fb8d4
	int nDstStepY;
Packit 1fb8d4
	INT16* HL, *LH;
Packit 1fb8d4
	INT16* HH, *LL;
Packit 1fb8d4
	INT16* L, *H, *LLx;
Packit 1fb8d4
	INT16* pLowBand[3];
Packit 1fb8d4
	INT16* pHighBand[3];
Packit 1fb8d4
	INT16* pDstBand[3];
Packit 1fb8d4
	int nLowStep[3];
Packit 1fb8d4
	int nHighStep[3];
Packit 1fb8d4
	int nDstStep[3];
Packit 1fb8d4
	int nLowCount[3];
Packit 1fb8d4
	int nHighCount[3];
Packit 1fb8d4
	int nDstCount[3];
Packit 1fb8d4
	nBandL = progressive_rfx_get_band_l_count(level);
Packit 1fb8d4
	nBandH = progressive_rfx_get_band_h_count(level);
Packit 1fb8d4
	offset = 0;
Packit 1fb8d4
	HL = &buffer[offset];
Packit 1fb8d4
	offset += (nBandH * nBandL);
Packit 1fb8d4
	LH = &buffer[offset];
Packit 1fb8d4
	offset += (nBandL * nBandH);
Packit 1fb8d4
	HH = &buffer[offset];
Packit 1fb8d4
	offset += (nBandH * nBandH);
Packit 1fb8d4
	LL = &buffer[offset];
Packit 1fb8d4
	nDstStepX = (nBandL + nBandH);
Packit 1fb8d4
	nDstStepY = (nBandL + nBandH);
Packit 1fb8d4
	offset = 0;
Packit 1fb8d4
	L = &temp[offset];
Packit 1fb8d4
	offset += (nBandL * nDstStepX);
Packit 1fb8d4
	H = &temp[offset];
Packit 1fb8d4
	LLx = &buffer[0];
Packit 1fb8d4
	/* horizontal (LL + HL -> L) */
Packit 1fb8d4
	pLowBand[0] = LL;
Packit 1fb8d4
	nLowStep[0] = nBandL;
Packit 1fb8d4
	pHighBand[0] = HL;
Packit 1fb8d4
	nHighStep[0] = nBandH;
Packit 1fb8d4
	pDstBand[0] = L;
Packit 1fb8d4
	nDstStep[0] = nDstStepX;
Packit 1fb8d4
	nLowCount[0] = nBandL;
Packit 1fb8d4
	nHighCount[0] = nBandH;
Packit 1fb8d4
	nDstCount[0] = nBandL;
Packit 1fb8d4
	progressive_rfx_idwt_x(pLowBand[0], nLowStep[0], pHighBand[0], nHighStep[0],
Packit 1fb8d4
	                       pDstBand[0], nDstStep[0], nLowCount[0], nHighCount[0], nDstCount[0]);
Packit 1fb8d4
	/* horizontal (LH + HH -> H) */
Packit 1fb8d4
	pLowBand[1] = LH;
Packit 1fb8d4
	nLowStep[1] = nBandL;
Packit 1fb8d4
	pHighBand[1] = HH;
Packit 1fb8d4
	nHighStep[1] = nBandH;
Packit 1fb8d4
	pDstBand[1] = H;
Packit 1fb8d4
	nDstStep[1] = nDstStepX;
Packit 1fb8d4
	nLowCount[1] = nBandL;
Packit 1fb8d4
	nHighCount[1] = nBandH;
Packit 1fb8d4
	nDstCount[1] = nBandH;
Packit 1fb8d4
	progressive_rfx_idwt_x(pLowBand[1], nLowStep[1], pHighBand[1], nHighStep[1],
Packit 1fb8d4
	                       pDstBand[1], nDstStep[1], nLowCount[1], nHighCount[1], nDstCount[1]);
Packit 1fb8d4
	/* vertical (L + H -> LL) */
Packit 1fb8d4
	pLowBand[2] = pDstBand[0];
Packit 1fb8d4
	nLowStep[2] = nDstStep[0];
Packit 1fb8d4
	pHighBand[2] = pDstBand[1];
Packit 1fb8d4
	nHighStep[2] = nDstStep[1];
Packit 1fb8d4
	pDstBand[2] = LLx;
Packit 1fb8d4
	nDstStep[2] = nDstStepY;
Packit 1fb8d4
	nLowCount[2] = nBandL;
Packit 1fb8d4
	nHighCount[2] = nBandH;
Packit 1fb8d4
	nDstCount[2] = nBandL + nBandH;
Packit 1fb8d4
	progressive_rfx_idwt_y(pLowBand[2], nLowStep[2], pHighBand[2], nHighStep[2],
Packit 1fb8d4
	                       pDstBand[2], nDstStep[2], nLowCount[2], nHighCount[2], nDstCount[2]);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_dwt_2d_decode(INT16* buffer, INT16* temp,
Packit 1fb8d4
        INT16* current, INT16* sign, BOOL diff)
Packit 1fb8d4
{
Packit 1fb8d4
	const primitives_t* prims = primitives_get();
Packit 1fb8d4
Packit 1fb8d4
	if (diff)
Packit 1fb8d4
		prims->add_16s(buffer, current, buffer, 4096);
Packit 1fb8d4
Packit 1fb8d4
	CopyMemory(current, buffer, 4096 * 2);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[3807], temp, 3);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[3007], temp, 2);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[0], temp, 1);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE void progressive_rfx_decode_block(const primitives_t* prims,
Packit 1fb8d4
        INT16* buffer,
Packit 1fb8d4
        int length, UINT32 shift)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!shift)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	prims->lShiftC_16s(buffer, shift, buffer, length);
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_decode_component(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* shift,
Packit 1fb8d4
        const BYTE* data, int length,
Packit 1fb8d4
        INT16* buffer, INT16* current,
Packit 1fb8d4
        INT16* sign, BOOL diff)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
	INT16* temp;
Packit 1fb8d4
	const primitives_t* prims = primitives_get();
Packit 1fb8d4
	status = rfx_rlgr_decode(RLGR1, data, length, buffer, 4096);
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
		return status;
Packit 1fb8d4
Packit 1fb8d4
	CopyMemory(sign, buffer, 4096 * 2);
Packit 1fb8d4
	rfx_differential_decode(&buffer[4015], 81); /* LL3 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[0], 1023, shift->HL1); /* HL1 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[1023], 1023, shift->LH1); /* LH1 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[2046], 961, shift->HH1); /* HH1 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3007], 272, shift->HL2); /* HL2 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3279], 272, shift->LH2); /* LH2 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3551], 256, shift->HH2); /* HH2 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3807], 72, shift->HL3); /* HL3 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3879], 72, shift->LH3); /* LH3 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[3951], 64, shift->HH3); /* HH3 */
Packit 1fb8d4
	progressive_rfx_decode_block(prims, &buffer[4015], 81, shift->LL3); /* LL3 */
Packit 1fb8d4
	temp = (INT16*) BufferPool_Take(progressive->bufferPool, -1); /* DWT buffer */
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode(buffer, temp, current, sign, diff);
Packit 1fb8d4
	BufferPool_Return(progressive->bufferPool, temp);
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_decompress_tile_first(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        RFX_PROGRESSIVE_TILE* tile)
Packit 1fb8d4
{
Packit 1fb8d4
	BOOL diff;
Packit 1fb8d4
	BYTE* pBuffer;
Packit 1fb8d4
	INT16* pSign[3];
Packit 1fb8d4
	INT16* pSrcDst[3];
Packit 1fb8d4
	INT16* pCurrent[3];
Packit 1fb8d4
	PROGRESSIVE_BLOCK_REGION* region;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftCr;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantCr;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgCr;
Packit 1fb8d4
	RFX_PROGRESSIVE_CODEC_QUANT* quantProgVal;
Packit 1fb8d4
	static const prim_size_t roi_64x64 = { 64, 64 };
Packit 1fb8d4
	const primitives_t* prims = primitives_get();
Packit 1fb8d4
Packit 1fb8d4
	tile->pass = 1;
Packit 1fb8d4
	diff = tile->flags & RFX_TILE_DIFFERENCE;
Packit 1fb8d4
	WLog_Print(progressive->log, WLOG_DEBUG,
Packit 1fb8d4
	           "ProgressiveTile%s: quantIdx Y: %"PRIu8" Cb: %"PRIu8" Cr: %"PRIu8" xIdx: %"PRIu16" yIdx: %"PRIu16" flags: 0x%02"PRIX8" quality: %"PRIu8" yLen: %"PRIu16" cbLen: %"PRIu16" crLen: %"PRIu16" tailLen: %"PRIu16"",
Packit 1fb8d4
	           (tile->blockType == PROGRESSIVE_WBT_TILE_FIRST) ? "First" : "Simple",
Packit 1fb8d4
	           tile->quantIdxY, tile->quantIdxCb, tile->quantIdxCr,
Packit 1fb8d4
	           tile->xIdx, tile->yIdx, tile->flags, tile->quality, tile->yLen,
Packit 1fb8d4
	           tile->cbLen, tile->crLen, tile->tailLen);
Packit 1fb8d4
	region = &(progressive->region);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxY >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantY = &(region->quantVals[tile->quantIdxY]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxCb >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantCb = &(region->quantVals[tile->quantIdxCb]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxCr >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantCr = &(region->quantVals[tile->quantIdxCr]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quality == 0xFF)
Packit 1fb8d4
	{
Packit 1fb8d4
		quantProgVal = &(progressive->quantProgValFull);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		if (tile->quality >= region->numProgQuant)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
Packit 1fb8d4
		quantProgVal = &(region->quantProgVals[tile->quality]);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	quantProgY = &(quantProgVal->yQuantValues);
Packit 1fb8d4
	quantProgCb = &(quantProgVal->cbQuantValues);
Packit 1fb8d4
	quantProgCr = &(quantProgVal->crQuantValues);
Packit 1fb8d4
Packit 1fb8d4
	CopyMemory(&(tile->yQuant), quantY, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->cbQuant), quantCb, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->crQuant), quantCr, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->yProgQuant), quantProgY, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->cbProgQuant), quantProgCb, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->crProgQuant), quantProgCr, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
Packit 1fb8d4
	progressive_rfx_quant_add(quantY, quantProgY, &(tile->yBitPos));
Packit 1fb8d4
	progressive_rfx_quant_add(quantCb, quantProgCb, &(tile->cbBitPos));
Packit 1fb8d4
	progressive_rfx_quant_add(quantCr, quantProgCr, &(tile->crBitPos));
Packit 1fb8d4
	progressive_rfx_quant_add(quantY, quantProgY, &shiftY);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftY, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
	progressive_rfx_quant_add(quantCb, quantProgCb, &shiftCb);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftCb, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
	progressive_rfx_quant_add(quantCr, quantProgCr, &shiftCr);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftCr, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
Packit 1fb8d4
	if (!tile->data)
Packit 1fb8d4
	{
Packit 1fb8d4
		tile->data = (BYTE*) _aligned_malloc(64 * 64 * 4, 16);
Packit 1fb8d4
		if (!tile->data)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!tile->sign)
Packit 1fb8d4
	{
Packit 1fb8d4
		tile->sign = (BYTE*) _aligned_malloc((8192 + 32) * 3, 16);
Packit 1fb8d4
		if (!tile->sign)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (!tile->current)
Packit 1fb8d4
	{
Packit 1fb8d4
		tile->current = (BYTE*) _aligned_malloc((8192 + 32) * 3, 16);
Packit 1fb8d4
		if (!tile->current)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = tile->sign;
Packit 1fb8d4
	pSign[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pSign[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pSign[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = tile->current;
Packit 1fb8d4
	pCurrent[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pCurrent[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pCurrent[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = (BYTE*) BufferPool_Take(progressive->bufferPool, -1);
Packit 1fb8d4
	pSrcDst[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pSrcDst[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pSrcDst[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	progressive_rfx_decode_component(progressive, &shiftY, tile->yData, tile->yLen,
Packit 1fb8d4
	                                 pSrcDst[0], pCurrent[0], pSign[0], diff); /* Y */
Packit 1fb8d4
	progressive_rfx_decode_component(progressive, &shiftCb, tile->cbData, tile->cbLen,
Packit 1fb8d4
	                                 pSrcDst[1], pCurrent[1], pSign[1], diff); /* Cb */
Packit 1fb8d4
	progressive_rfx_decode_component(progressive, &shiftCr, tile->crData, tile->crLen,
Packit 1fb8d4
	                                 pSrcDst[2], pCurrent[2], pSign[2], diff); /* Cr */
Packit 1fb8d4
Packit 1fb8d4
	prims->yCbCrToRGB_16s8u_P3AC4R((const INT16**) pSrcDst, 64 * 2, tile->data, tile->stride,
Packit 1fb8d4
								tile->format, &roi_64x64);
Packit 1fb8d4
	BufferPool_Return(progressive->bufferPool, pBuffer);
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE INT16 progressive_rfx_srl_read(RFX_PROGRESSIVE_UPGRADE_STATE* state,
Packit 1fb8d4
        UINT32 numBits)
Packit 1fb8d4
{
Packit 1fb8d4
	int k;
Packit 1fb8d4
	UINT32 bit;
Packit 1fb8d4
	UINT32 max;
Packit 1fb8d4
	UINT32 mag;
Packit 1fb8d4
	UINT32 sign;
Packit 1fb8d4
	wBitStream* bs = state->srl;
Packit 1fb8d4
Packit 1fb8d4
	if (state->nz)
Packit 1fb8d4
	{
Packit 1fb8d4
		state->nz--;
Packit 1fb8d4
		return 0;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	k = state->kp / 8;
Packit 1fb8d4
Packit 1fb8d4
	if (!state->mode)
Packit 1fb8d4
	{
Packit 1fb8d4
		/* zero encoding */
Packit 1fb8d4
		bit = (bs->accumulator & 0x80000000) ? 1 : 0;
Packit 1fb8d4
		BitStream_Shift(bs, 1);
Packit 1fb8d4
Packit 1fb8d4
		if (!bit)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* '0' bit, nz >= (1 << k), nz = (1 << k) */
Packit 1fb8d4
			state->nz = (1 << k);
Packit 1fb8d4
			state->kp += 4;
Packit 1fb8d4
Packit 1fb8d4
			if (state->kp > 80)
Packit 1fb8d4
				state->kp = 80;
Packit 1fb8d4
Packit 1fb8d4
			state->nz--;
Packit 1fb8d4
			return 0;
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			/* '1' bit, nz < (1 << k), nz = next k bits */
Packit 1fb8d4
			state->nz = 0;
Packit 1fb8d4
			state->mode = 1; /* unary encoding is next */
Packit 1fb8d4
Packit 1fb8d4
			if (k)
Packit 1fb8d4
			{
Packit 1fb8d4
				bs->mask = ((1 << k) - 1);
Packit 1fb8d4
				state->nz = ((bs->accumulator >> (32 - k)) & bs->mask);
Packit 1fb8d4
				BitStream_Shift(bs, k);
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (state->nz)
Packit 1fb8d4
			{
Packit 1fb8d4
				state->nz--;
Packit 1fb8d4
				return 0;
Packit 1fb8d4
			}
Packit 1fb8d4
		}
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	state->mode = 0; /* zero encoding is next */
Packit 1fb8d4
	/* unary encoding */
Packit 1fb8d4
	/* read sign bit */
Packit 1fb8d4
	sign = (bs->accumulator & 0x80000000) ? 1 : 0;
Packit 1fb8d4
	BitStream_Shift(bs, 1);
Packit 1fb8d4
	state->kp -= 6;
Packit 1fb8d4
Packit 1fb8d4
	if (state->kp < 0)
Packit 1fb8d4
		state->kp = 0;
Packit 1fb8d4
Packit 1fb8d4
	if (numBits == 1)
Packit 1fb8d4
		return sign ? -1 : 1;
Packit 1fb8d4
Packit 1fb8d4
	mag = 1;
Packit 1fb8d4
	max = (1 << numBits) - 1;
Packit 1fb8d4
Packit 1fb8d4
	while (mag < max)
Packit 1fb8d4
	{
Packit 1fb8d4
		bit = (bs->accumulator & 0x80000000) ? 1 : 0;
Packit 1fb8d4
		BitStream_Shift(bs, 1);
Packit 1fb8d4
Packit 1fb8d4
		if (bit)
Packit 1fb8d4
			break;
Packit 1fb8d4
Packit 1fb8d4
		mag++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return sign ? -1 * mag : mag;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_upgrade_state_finish(RFX_PROGRESSIVE_UPGRADE_STATE* state)
Packit 1fb8d4
{
Packit 1fb8d4
	int pad;
Packit 1fb8d4
	wBitStream* srl;
Packit 1fb8d4
	wBitStream* raw;
Packit 1fb8d4
	srl = state->srl;
Packit 1fb8d4
	raw = state->raw;
Packit 1fb8d4
	/* Read trailing bits from RAW/SRL bit streams */
Packit 1fb8d4
	pad = (raw->position % 8) ? (8 - (raw->position % 8)) : 0;
Packit 1fb8d4
Packit 1fb8d4
	if (pad)
Packit 1fb8d4
		BitStream_Shift(raw, pad);
Packit 1fb8d4
Packit 1fb8d4
	pad = (srl->position % 8) ? (8 - (srl->position % 8)) : 0;
Packit 1fb8d4
Packit 1fb8d4
	if (pad)
Packit 1fb8d4
		BitStream_Shift(srl, pad);
Packit 1fb8d4
Packit 1fb8d4
	if (BitStream_GetRemainingLength(srl) == 8)
Packit 1fb8d4
		BitStream_Shift(srl, 8);
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_upgrade_block(RFX_PROGRESSIVE_UPGRADE_STATE* state,
Packit 1fb8d4
        INT16* buffer,	INT16* sign, UINT32 length,
Packit 1fb8d4
        UINT32 shift, UINT32 bitPos, UINT32 numBits)
Packit 1fb8d4
{
Packit 1fb8d4
	int index;
Packit 1fb8d4
	INT16 input;
Packit 1fb8d4
	wBitStream* raw;
Packit 1fb8d4
Packit 1fb8d4
	if (!numBits)
Packit 1fb8d4
		return 1;
Packit 1fb8d4
Packit 1fb8d4
	raw = state->raw;
Packit 1fb8d4
Packit 1fb8d4
	if (!state->nonLL)
Packit 1fb8d4
	{
Packit 1fb8d4
		for (index = 0; index < length; index++)
Packit 1fb8d4
		{
Packit 1fb8d4
			raw->mask = ((1 << numBits) - 1);
Packit 1fb8d4
			input = (INT16)((raw->accumulator >> (32 - numBits)) & raw->mask);
Packit 1fb8d4
			BitStream_Shift(raw, numBits);
Packit 1fb8d4
			buffer[index] += (input << shift);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		return 1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < length; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		if (sign[index] > 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* sign > 0, read from raw */
Packit 1fb8d4
			raw->mask = ((1 << numBits) - 1);
Packit 1fb8d4
			input = (INT16)((raw->accumulator >> (32 - numBits)) & raw->mask);
Packit 1fb8d4
			BitStream_Shift(raw, numBits);
Packit 1fb8d4
		}
Packit 1fb8d4
		else if (sign[index] < 0)
Packit 1fb8d4
		{
Packit 1fb8d4
			/* sign < 0, read from raw */
Packit 1fb8d4
			raw->mask = ((1 << numBits) - 1);
Packit 1fb8d4
			input = (INT16)((raw->accumulator >> (32 - numBits)) & raw->mask);
Packit 1fb8d4
			BitStream_Shift(raw, numBits);
Packit 1fb8d4
			input *= -1;
Packit 1fb8d4
		}
Packit 1fb8d4
		else
Packit 1fb8d4
		{
Packit 1fb8d4
			/* sign == 0, read from srl */
Packit 1fb8d4
			input = progressive_rfx_srl_read(state, numBits);
Packit 1fb8d4
			sign[index] = input;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		buffer[index] += (input << shift);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_rfx_upgrade_component(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* shift,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* bitPos,
Packit 1fb8d4
        RFX_COMPONENT_CODEC_QUANT* numBits,
Packit 1fb8d4
        INT16* buffer,
Packit 1fb8d4
        INT16* current, INT16* sign,
Packit 1fb8d4
        const BYTE* srlData,
Packit 1fb8d4
        UINT32 srlLen, const BYTE* rawData,
Packit 1fb8d4
        UINT32 rawLen)
Packit 1fb8d4
{
Packit 1fb8d4
	INT16* temp;
Packit 1fb8d4
	UINT32 aRawLen;
Packit 1fb8d4
	UINT32 aSrlLen;
Packit 1fb8d4
	wBitStream s_srl;
Packit 1fb8d4
	wBitStream s_raw;
Packit 1fb8d4
	RFX_PROGRESSIVE_UPGRADE_STATE state;
Packit 1fb8d4
Packit 1fb8d4
	ZeroMemory(&s_srl, sizeof(wBitStream));
Packit 1fb8d4
	ZeroMemory(&s_raw, sizeof(wBitStream));
Packit 1fb8d4
	ZeroMemory(&state, sizeof(RFX_PROGRESSIVE_UPGRADE_STATE));
Packit 1fb8d4
	state.kp = 8;
Packit 1fb8d4
	state.mode = 0;
Packit 1fb8d4
	state.srl = &s_srl;
Packit 1fb8d4
	state.raw = &s_raw;
Packit 1fb8d4
	BitStream_Attach(state.srl, srlData, srlLen);
Packit 1fb8d4
	BitStream_Fetch(state.srl);
Packit 1fb8d4
	BitStream_Attach(state.raw, rawData, rawLen);
Packit 1fb8d4
	BitStream_Fetch(state.raw);
Packit 1fb8d4
Packit 1fb8d4
	state.nonLL = TRUE;
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[0], &sign[0], 1023, shift->HL1,
Packit 1fb8d4
	                              bitPos->HL1, numBits->HL1); /* HL1 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[1023], &sign[1023], 1023,
Packit 1fb8d4
	                              shift->LH1, bitPos->LH1, numBits->LH1); /* LH1 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[2046], &sign[2046], 961,
Packit 1fb8d4
	                              shift->HH1, bitPos->HH1, numBits->HH1); /* HH1 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3007], &sign[3007], 272,
Packit 1fb8d4
	                              shift->HL2, bitPos->HL2, numBits->HL2); /* HL2 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3279], &sign[3279], 272,
Packit 1fb8d4
	                              shift->LH2, bitPos->LH2, numBits->LH2); /* LH2 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3551], &sign[3551], 256,
Packit 1fb8d4
	                              shift->HH2, bitPos->HH2, numBits->HH2); /* HH2 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3807], &sign[3807], 72,
Packit 1fb8d4
	                              shift->HL3, bitPos->HL3, numBits->HL3); /* HL3 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3879], &sign[3879], 72,
Packit 1fb8d4
	                              shift->LH3, bitPos->LH3, numBits->LH3); /* LH3 */
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[3951], &sign[3951], 64,
Packit 1fb8d4
	                              shift->HH3, bitPos->HH3, numBits->HH3); /* HH3 */
Packit 1fb8d4
Packit 1fb8d4
	state.nonLL = FALSE;
Packit 1fb8d4
	progressive_rfx_upgrade_block(&state, &current[4015], &sign[4015], 81,
Packit 1fb8d4
	                              shift->LL3, bitPos->LL3, numBits->LL3); /* LL3 */
Packit 1fb8d4
	progressive_rfx_upgrade_state_finish(&state);
Packit 1fb8d4
	aRawLen = (state.raw->position + 7) / 8;
Packit 1fb8d4
	aSrlLen = (state.srl->position + 7) / 8;
Packit 1fb8d4
Packit 1fb8d4
	if ((aRawLen != rawLen) || (aSrlLen != srlLen))
Packit 1fb8d4
	{
Packit 1fb8d4
		int pRawLen = 0;
Packit 1fb8d4
		int pSrlLen = 0;
Packit 1fb8d4
Packit 1fb8d4
		if (rawLen)
Packit 1fb8d4
			pRawLen = (int)((((float) aRawLen) / ((float) rawLen)) * 100.0f);
Packit 1fb8d4
Packit 1fb8d4
		if (srlLen)
Packit 1fb8d4
			pSrlLen = (int)((((float) aSrlLen) / ((float) srlLen)) * 100.0f);
Packit 1fb8d4
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_INFO,
Packit 1fb8d4
		           "RAW: %"PRIu32"/%"PRIu32" %d%% (%"PRIu32"/%"PRIu32":%"PRIu32")\tSRL: %"PRIu32"/%"PRIu32" %d%% (%"PRIu32"/%"PRIu32":%"PRIu32")",
Packit 1fb8d4
		           aRawLen, rawLen, pRawLen, state.raw->position, rawLen * 8,
Packit 1fb8d4
		           (rawLen * 8) - state.raw->position,
Packit 1fb8d4
		           aSrlLen, srlLen, pSrlLen, state.srl->position, srlLen * 8,
Packit 1fb8d4
		           (srlLen * 8) - state.srl->position);
Packit 1fb8d4
		return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	temp = (INT16*) BufferPool_Take(progressive->bufferPool, -1); /* DWT buffer */
Packit 1fb8d4
	CopyMemory(buffer, current, 4096 * 2);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[3807], temp, 3);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[3007], temp, 2);
Packit 1fb8d4
	progressive_rfx_dwt_2d_decode_block(&buffer[0], temp, 1);
Packit 1fb8d4
	BufferPool_Return(progressive->bufferPool, temp);
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_decompress_tile_upgrade(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        RFX_PROGRESSIVE_TILE* tile)
Packit 1fb8d4
{
Packit 1fb8d4
	int status;
Packit 1fb8d4
	BYTE* pBuffer;
Packit 1fb8d4
	INT16* pSign[3];
Packit 1fb8d4
	INT16* pSrcDst[3];
Packit 1fb8d4
	INT16* pCurrent[3];
Packit 1fb8d4
	PROGRESSIVE_BLOCK_REGION* region;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT shiftCr;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT yBitPos;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT cbBitPos;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT crBitPos;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT yNumBits;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT cbNumBits;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT crNumBits;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantCr;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgY;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgCb;
Packit 1fb8d4
	RFX_COMPONENT_CODEC_QUANT* quantProgCr;
Packit 1fb8d4
	RFX_PROGRESSIVE_CODEC_QUANT* quantProg;
Packit 1fb8d4
	static const prim_size_t roi_64x64 = { 64, 64 };
Packit 1fb8d4
	const primitives_t* prims = primitives_get();
Packit 1fb8d4
	tile->pass++;
Packit 1fb8d4
	WLog_Print(progressive->log, WLOG_DEBUG,
Packit 1fb8d4
	           "ProgressiveTileUpgrade: pass: %"PRIu16" quantIdx Y: %"PRIu8" Cb: %"PRIu8" Cr: %"PRIu8" xIdx: %"PRIu16" yIdx: %"PRIu16" quality: %"PRIu8" ySrlLen: %"PRIu16" yRawLen: %"PRIu16" cbSrlLen: %"PRIu16" cbRawLen: %"PRIu16" crSrlLen: %"PRIu16" crRawLen: %"PRIu16"",
Packit 1fb8d4
	           tile->pass, tile->quantIdxY, tile->quantIdxCb, tile->quantIdxCr, tile->xIdx,
Packit 1fb8d4
	           tile->yIdx, tile->quality, tile->ySrlLen, tile->yRawLen, tile->cbSrlLen,
Packit 1fb8d4
	           tile->cbRawLen, tile->crSrlLen, tile->crRawLen);
Packit 1fb8d4
	region = &(progressive->region);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxY >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantY = &(region->quantVals[tile->quantIdxY]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxCb >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantCb = &(region->quantVals[tile->quantIdxCb]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quantIdxCr >= region->numQuant)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	quantCr = &(region->quantVals[tile->quantIdxCr]);
Packit 1fb8d4
Packit 1fb8d4
	if (tile->quality == 0xFF)
Packit 1fb8d4
	{
Packit 1fb8d4
		quantProg = &(progressive->quantProgValFull);
Packit 1fb8d4
	}
Packit 1fb8d4
	else
Packit 1fb8d4
	{
Packit 1fb8d4
		if (tile->quality >= region->numProgQuant)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
Packit 1fb8d4
		quantProg = &(region->quantProgVals[tile->quality]);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	quantProgY = &(quantProg->yQuantValues);
Packit 1fb8d4
	quantProgCb = &(quantProg->cbQuantValues);
Packit 1fb8d4
	quantProgCr = &(quantProg->crQuantValues);
Packit 1fb8d4
Packit 1fb8d4
	if (!progressive_rfx_quant_cmp_equal(quantY, &(tile->yQuant)))
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_WARN, "non-progressive quantY has changed!");
Packit 1fb8d4
Packit 1fb8d4
	if (!progressive_rfx_quant_cmp_equal(quantCb, &(tile->cbQuant)))
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_WARN, "non-progressive quantCb has changed!");
Packit 1fb8d4
Packit 1fb8d4
	if (!progressive_rfx_quant_cmp_equal(quantCr, &(tile->crQuant)))
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_WARN, "non-progressive quantCr has changed!");
Packit 1fb8d4
Packit 1fb8d4
	progressive_rfx_quant_add(quantY, quantProgY, &yBitPos);
Packit 1fb8d4
	progressive_rfx_quant_add(quantCb, quantProgCb, &cbBitPos);
Packit 1fb8d4
	progressive_rfx_quant_add(quantCr, quantProgCr, &crBitPos);
Packit 1fb8d4
	progressive_rfx_quant_sub(&(tile->yBitPos), &yBitPos, &yNumBits);
Packit 1fb8d4
	progressive_rfx_quant_sub(&(tile->cbBitPos), &cbBitPos, &cbNumBits);
Packit 1fb8d4
	progressive_rfx_quant_sub(&(tile->crBitPos), &crBitPos, &crNumBits);
Packit 1fb8d4
	progressive_rfx_quant_add(quantY, quantProgY, &shiftY);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftY, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
	progressive_rfx_quant_add(quantCb, quantProgCb, &shiftCb);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftCb, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
	progressive_rfx_quant_add(quantCr, quantProgCr, &shiftCr);
Packit 1fb8d4
	progressive_rfx_quant_lsub(&shiftCr, 1); /* -6 + 5 = -1 */
Packit 1fb8d4
Packit 1fb8d4
	CopyMemory(&(tile->yBitPos), &yBitPos, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->cbBitPos), &cbBitPos, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->crBitPos), &crBitPos, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->yQuant), quantY, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->cbQuant), quantCb, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->crQuant), quantCr, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->yProgQuant), quantProgY, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->cbProgQuant), quantProgCb, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
	CopyMemory(&(tile->crProgQuant), quantProgCr, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = tile->sign;
Packit 1fb8d4
	pSign[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pSign[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pSign[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = tile->current;
Packit 1fb8d4
	pCurrent[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pCurrent[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pCurrent[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	pBuffer = (BYTE*) BufferPool_Take(progressive->bufferPool, -1);
Packit 1fb8d4
	pSrcDst[0] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 0) + 16])); /* Y/R buffer */
Packit 1fb8d4
	pSrcDst[1] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 1) + 16])); /* Cb/G buffer */
Packit 1fb8d4
	pSrcDst[2] = (INT16*)((BYTE*)(&pBuffer[((8192 + 32) * 2) + 16])); /* Cr/B buffer */
Packit 1fb8d4
Packit 1fb8d4
	status = progressive_rfx_upgrade_component(progressive, &shiftY, quantProgY,
Packit 1fb8d4
	         &yNumBits,
Packit 1fb8d4
	         pSrcDst[0], pCurrent[0], pSign[0], tile->ySrlData, tile->ySrlLen,
Packit 1fb8d4
	         tile->yRawData, tile->yRawLen); /* Y */
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	status = progressive_rfx_upgrade_component(progressive, &shiftCb, quantProgCb,
Packit 1fb8d4
	         &cbNumBits,
Packit 1fb8d4
	         pSrcDst[1], pCurrent[1], pSign[1], tile->cbSrlData, tile->cbSrlLen,
Packit 1fb8d4
	         tile->cbRawData, tile->cbRawLen); /* Cb */
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	status = progressive_rfx_upgrade_component(progressive, &shiftCr, quantProgCr,
Packit 1fb8d4
	         &crNumBits,
Packit 1fb8d4
	         pSrcDst[2], pCurrent[2], pSign[2], tile->crSrlData, tile->crSrlLen,
Packit 1fb8d4
	         tile->crRawData, tile->crRawLen); /* Cr */
Packit 1fb8d4
Packit 1fb8d4
	if (status < 0)
Packit 1fb8d4
		return -1;
Packit 1fb8d4
Packit 1fb8d4
	prims->yCbCrToRGB_16s8u_P3AC4R((const INT16**) pSrcDst, 64 * 2,
Packit 1fb8d4
	                               tile->data, tile->stride, tile->format,
Packit 1fb8d4
	                               &roi_64x64);
Packit 1fb8d4
	BufferPool_Return(progressive->bufferPool, pBuffer);
Packit 1fb8d4
	return 1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
static INLINE int progressive_process_tiles(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
        const BYTE* blocks, UINT32 blocksLen,
Packit 1fb8d4
        const PROGRESSIVE_SURFACE_CONTEXT* surface)
Packit 1fb8d4
{
Packit 1fb8d4
	int status = -1;
Packit 1fb8d4
	const BYTE* block;
Packit 1fb8d4
	UINT16 xIdx;
Packit 1fb8d4
	UINT16 yIdx;
Packit 1fb8d4
	UINT16 zIdx;
Packit 1fb8d4
	UINT16 index;
Packit 1fb8d4
	UINT32 boffset;
Packit 1fb8d4
	UINT16 blockType;
Packit 1fb8d4
	UINT32 blockLen;
Packit 1fb8d4
	UINT32 count = 0;
Packit 1fb8d4
	UINT32 offset = 0;
Packit 1fb8d4
	RFX_PROGRESSIVE_TILE* tile;
Packit 1fb8d4
	RFX_PROGRESSIVE_TILE** tiles;
Packit 1fb8d4
	PROGRESSIVE_BLOCK_REGION* region;
Packit 1fb8d4
	region = &(progressive->region);
Packit 1fb8d4
	tiles = region->tiles;
Packit 1fb8d4
Packit 1fb8d4
	while ((blocksLen - offset) >= 6)
Packit 1fb8d4
	{
Packit 1fb8d4
		boffset = 0;
Packit 1fb8d4
		block = &blocks[offset];
Packit 1fb8d4
		blockType = *((UINT16*) &block[boffset + 0]); /* blockType (2 bytes) */
Packit 1fb8d4
		blockLen = *((UINT32*) &block[boffset + 2]); /* blockLen (4 bytes) */
Packit 1fb8d4
		boffset += 6;
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_DEBUG, "%s", progressive_get_block_type_string(blockType));
Packit 1fb8d4
Packit 1fb8d4
		if ((blocksLen - offset) < blockLen)
Packit 1fb8d4
			return -1003;
Packit 1fb8d4
Packit 1fb8d4
		switch (blockType)
Packit 1fb8d4
		{
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_SIMPLE:
Packit 1fb8d4
				if ((blockLen - boffset) < 16)
Packit 1fb8d4
					return -1022;
Packit 1fb8d4
Packit 1fb8d4
				xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				zIdx = (yIdx * surface->gridWidth) + xIdx;
Packit 1fb8d4
Packit 1fb8d4
				if (zIdx >= surface->gridSize)
Packit 1fb8d4
					return -1;
Packit 1fb8d4
Packit 1fb8d4
				tiles[count] = tile = &(surface->tiles[zIdx]);
Packit 1fb8d4
				tile->blockType = blockType;
Packit 1fb8d4
				tile->blockLen = blockLen;
Packit 1fb8d4
				tile->quality = 0xFF; /* simple tiles use no progressive techniques */
Packit 1fb8d4
				tile->quantIdxY = block[boffset + 0]; /* quantIdxY (1 byte) */
Packit 1fb8d4
				tile->quantIdxCb = block[boffset + 1]; /* quantIdxCb (1 byte) */
Packit 1fb8d4
				tile->quantIdxCr = block[boffset + 2]; /* quantIdxCr (1 byte) */
Packit 1fb8d4
				tile->xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				tile->yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				tile->flags = block[boffset + 7] & 1; /* flags (1 byte) */
Packit 1fb8d4
				tile->yLen = *((UINT16*) &block[boffset + 8]); /* yLen (2 bytes) */
Packit 1fb8d4
				tile->cbLen = *((UINT16*) &block[boffset + 10]); /* cbLen (2 bytes) */
Packit 1fb8d4
				tile->crLen = *((UINT16*) &block[boffset + 12]); /* crLen (2 bytes) */
Packit 1fb8d4
				tile->tailLen = *((UINT16*) &block[boffset + 14]); /* tailLen (2 bytes) */
Packit 1fb8d4
				boffset += 16;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->yLen)
Packit 1fb8d4
					return -1023;
Packit 1fb8d4
Packit 1fb8d4
				tile->yData = &block[boffset];
Packit 1fb8d4
				boffset += tile->yLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->cbLen)
Packit 1fb8d4
					return -1024;
Packit 1fb8d4
Packit 1fb8d4
				tile->cbData = &block[boffset];
Packit 1fb8d4
				boffset += tile->cbLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->crLen)
Packit 1fb8d4
					return -1025;
Packit 1fb8d4
Packit 1fb8d4
				tile->crData = &block[boffset];
Packit 1fb8d4
				boffset += tile->crLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->tailLen)
Packit 1fb8d4
					return -1026;
Packit 1fb8d4
Packit 1fb8d4
				tile->tailData = &block[boffset];
Packit 1fb8d4
				boffset += tile->tailLen;
Packit 1fb8d4
				tile->width = 64;
Packit 1fb8d4
				tile->height = 64;
Packit 1fb8d4
				tile->x = tile->xIdx * 64;
Packit 1fb8d4
				tile->y = tile->yIdx * 64;
Packit 1fb8d4
				tile->format = progressive->format;
Packit 1fb8d4
				tile->stride = GetBytesPerPixel(tile->format) * tile->width;
Packit 1fb8d4
				tile->flags &= 1;
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_FIRST:
Packit 1fb8d4
				if ((blockLen - boffset) < 17)
Packit 1fb8d4
					return -1027;
Packit 1fb8d4
Packit 1fb8d4
				xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				zIdx = (yIdx * surface->gridWidth) + xIdx;
Packit 1fb8d4
Packit 1fb8d4
				if (zIdx >= surface->gridSize)
Packit 1fb8d4
					return -1;
Packit 1fb8d4
Packit 1fb8d4
				tiles[count] = tile = &(surface->tiles[zIdx]);
Packit 1fb8d4
				tile->blockType = blockType;
Packit 1fb8d4
				tile->blockLen = blockLen;
Packit 1fb8d4
				tile->quantIdxY = block[boffset + 0]; /* quantIdxY (1 byte) */
Packit 1fb8d4
				tile->quantIdxCb = block[boffset + 1]; /* quantIdxCb (1 byte) */
Packit 1fb8d4
				tile->quantIdxCr = block[boffset + 2]; /* quantIdxCr (1 byte) */
Packit 1fb8d4
				tile->xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				tile->yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				tile->flags = block[boffset + 7] & 1; /* flags (1 byte) */
Packit 1fb8d4
				tile->quality = block[boffset + 8]; /* quality (1 byte) */
Packit 1fb8d4
				tile->yLen = *((UINT16*) &block[boffset + 9]); /* yLen (2 bytes) */
Packit 1fb8d4
				tile->cbLen = *((UINT16*) &block[boffset + 11]); /* cbLen (2 bytes) */
Packit 1fb8d4
				tile->crLen = *((UINT16*) &block[boffset + 13]); /* crLen (2 bytes) */
Packit 1fb8d4
				tile->tailLen = *((UINT16*) &block[boffset + 15]); /* tailLen (2 bytes) */
Packit 1fb8d4
				boffset += 17;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->yLen)
Packit 1fb8d4
					return -1028;
Packit 1fb8d4
Packit 1fb8d4
				tile->yData = &block[boffset];
Packit 1fb8d4
				boffset += tile->yLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->cbLen)
Packit 1fb8d4
					return -1029;
Packit 1fb8d4
Packit 1fb8d4
				tile->cbData = &block[boffset];
Packit 1fb8d4
				boffset += tile->cbLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->crLen)
Packit 1fb8d4
					return -1030;
Packit 1fb8d4
Packit 1fb8d4
				tile->crData = &block[boffset];
Packit 1fb8d4
				boffset += tile->crLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->tailLen)
Packit 1fb8d4
					return -1031;
Packit 1fb8d4
Packit 1fb8d4
				tile->tailData = &block[boffset];
Packit 1fb8d4
				boffset += tile->tailLen;
Packit 1fb8d4
				tile->width = 64;
Packit 1fb8d4
				tile->height = 64;
Packit 1fb8d4
				tile->x = tile->xIdx * 64;
Packit 1fb8d4
				tile->y = tile->yIdx * 64;
Packit 1fb8d4
				tile->format = progressive->format;
Packit 1fb8d4
				tile->stride = GetBytesPerPixel(tile->format) * tile->width;
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_UPGRADE:
Packit 1fb8d4
				if ((blockLen - boffset) < 20)
Packit 1fb8d4
					return -1032;
Packit 1fb8d4
Packit 1fb8d4
				xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				zIdx = (yIdx * surface->gridWidth) + xIdx;
Packit 1fb8d4
Packit 1fb8d4
				if (zIdx >= surface->gridSize)
Packit 1fb8d4
					return -1;
Packit 1fb8d4
Packit 1fb8d4
				tiles[count] = tile = &(surface->tiles[zIdx]);
Packit 1fb8d4
				tile->blockType = blockType;
Packit 1fb8d4
				tile->blockLen = blockLen;
Packit 1fb8d4
				tile->flags = 0;
Packit 1fb8d4
				tile->quantIdxY = block[boffset + 0]; /* quantIdxY (1 byte) */
Packit 1fb8d4
				tile->quantIdxCb = block[boffset + 1]; /* quantIdxCb (1 byte) */
Packit 1fb8d4
				tile->quantIdxCr = block[boffset + 2]; /* quantIdxCr (1 byte) */
Packit 1fb8d4
				tile->xIdx = *((UINT16*) &block[boffset + 3]); /* xIdx (2 bytes) */
Packit 1fb8d4
				tile->yIdx = *((UINT16*) &block[boffset + 5]); /* yIdx (2 bytes) */
Packit 1fb8d4
				tile->quality = block[boffset + 7]; /* quality (1 byte) */
Packit 1fb8d4
				tile->ySrlLen = *((UINT16*) &block[boffset + 8]); /* ySrlLen (2 bytes) */
Packit 1fb8d4
				tile->yRawLen = *((UINT16*) &block[boffset + 10]); /* yRawLen (2 bytes) */
Packit 1fb8d4
				tile->cbSrlLen = *((UINT16*) &block[boffset + 12]); /* cbSrlLen (2 bytes) */
Packit 1fb8d4
				tile->cbRawLen = *((UINT16*) &block[boffset + 14]); /* cbRawLen (2 bytes) */
Packit 1fb8d4
				tile->crSrlLen = *((UINT16*) &block[boffset + 16]); /* crSrlLen (2 bytes) */
Packit 1fb8d4
				tile->crRawLen = *((UINT16*) &block[boffset + 18]); /* crRawLen (2 bytes) */
Packit 1fb8d4
				boffset += 20;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->ySrlLen)
Packit 1fb8d4
					return -1033;
Packit 1fb8d4
Packit 1fb8d4
				tile->ySrlData = &block[boffset];
Packit 1fb8d4
				boffset += tile->ySrlLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->yRawLen)
Packit 1fb8d4
					return -1034;
Packit 1fb8d4
Packit 1fb8d4
				tile->yRawData = &block[boffset];
Packit 1fb8d4
				boffset += tile->yRawLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->cbSrlLen)
Packit 1fb8d4
					return -1035;
Packit 1fb8d4
Packit 1fb8d4
				tile->cbSrlData = &block[boffset];
Packit 1fb8d4
				boffset += tile->cbSrlLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->cbRawLen)
Packit 1fb8d4
					return -1036;
Packit 1fb8d4
Packit 1fb8d4
				tile->cbRawData = &block[boffset];
Packit 1fb8d4
				boffset += tile->cbRawLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->crSrlLen)
Packit 1fb8d4
					return -1037;
Packit 1fb8d4
Packit 1fb8d4
				tile->crSrlData = &block[boffset];
Packit 1fb8d4
				boffset += tile->crSrlLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((tile->blockLen - boffset) < tile->crRawLen)
Packit 1fb8d4
					return -1038;
Packit 1fb8d4
Packit 1fb8d4
				tile->crRawData = &block[boffset];
Packit 1fb8d4
				boffset += tile->crRawLen;
Packit 1fb8d4
				tile->width = 64;
Packit 1fb8d4
				tile->height = 64;
Packit 1fb8d4
				tile->x = tile->xIdx * 64;
Packit 1fb8d4
				tile->y = tile->yIdx * 64;
Packit 1fb8d4
				tile->format = progressive->format;
Packit 1fb8d4
				tile->stride = GetBytesPerPixel(tile->format) * tile->width;
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			default:
Packit 1fb8d4
				return -1039;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (boffset != blockLen)
Packit 1fb8d4
			return -1040;
Packit 1fb8d4
Packit 1fb8d4
		offset += blockLen;
Packit 1fb8d4
		count++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (offset != blocksLen)
Packit 1fb8d4
		return -1041;
Packit 1fb8d4
Packit 1fb8d4
	if (count != region->numTiles)
Packit 1fb8d4
	{
Packit 1fb8d4
		WLog_Print(progressive->log, WLOG_WARN,
Packit 1fb8d4
		           "numTiles inconsistency: actual: %"PRIu32", expected: %"PRIu16"\n", count,
Packit 1fb8d4
		           region->numTiles);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (index = 0; index < region->numTiles; index++)
Packit 1fb8d4
	{
Packit 1fb8d4
		tile = tiles[index];
Packit 1fb8d4
Packit 1fb8d4
		switch (tile->blockType)
Packit 1fb8d4
		{
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_SIMPLE:
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_FIRST:
Packit 1fb8d4
				status = progressive_decompress_tile_first(progressive, tile);
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_TILE_UPGRADE:
Packit 1fb8d4
				status = progressive_decompress_tile_upgrade(progressive, tile);
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (status < 0)
Packit 1fb8d4
			return -1;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return (int) offset;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
INT32 progressive_decompress(PROGRESSIVE_CONTEXT* progressive,
Packit 1fb8d4
                             const BYTE* pSrcData, UINT32 SrcSize,
Packit 1fb8d4
                             BYTE* pDstData, UINT32 DstFormat,
Packit 1fb8d4
                             UINT32 nDstStep, UINT32 nXDst, UINT32 nYDst,
Packit 1fb8d4
                             REGION16* invalidRegion, UINT16 surfaceId)
Packit 1fb8d4
{
Packit 1fb8d4
	INT32 rc = 1;
Packit 1fb8d4
	INT32 status;
Packit 1fb8d4
	const BYTE* block;
Packit 1fb8d4
	const BYTE* blocks;
Packit 1fb8d4
	UINT16 i, j;
Packit 1fb8d4
	UINT16 index;
Packit 1fb8d4
	UINT16 boxLeft;
Packit 1fb8d4
	UINT16 boxTop;
Packit 1fb8d4
	UINT16 boxRight;
Packit 1fb8d4
	UINT16 boxBottom;
Packit 1fb8d4
	UINT16 idxLeft;
Packit 1fb8d4
	UINT16 idxTop;
Packit 1fb8d4
	UINT16 idxRight;
Packit 1fb8d4
	UINT16 idxBottom;
Packit 1fb8d4
	UINT32 boffset;
Packit 1fb8d4
	UINT16 blockType;
Packit 1fb8d4
	UINT32 blockLen;
Packit 1fb8d4
	UINT32 blocksLen;
Packit 1fb8d4
	UINT32 count = 0;
Packit 1fb8d4
	UINT32 offset = 0;
Packit 1fb8d4
	RFX_RECT* rect = NULL;
Packit 1fb8d4
	REGION16 clippingRects, updateRegion;
Packit 1fb8d4
	PROGRESSIVE_SURFACE_CONTEXT* surface;
Packit 1fb8d4
	PROGRESSIVE_BLOCK_REGION* region;
Packit 1fb8d4
	surface = (PROGRESSIVE_SURFACE_CONTEXT*) progressive_get_surface_data(
Packit 1fb8d4
	              progressive, surfaceId);
Packit 1fb8d4
Packit 1fb8d4
	if (!surface)
Packit 1fb8d4
		return -1001;
Packit 1fb8d4
Packit 1fb8d4
	blocks = pSrcData;
Packit 1fb8d4
	blocksLen = SrcSize;
Packit 1fb8d4
	region = &(progressive->region);
Packit 1fb8d4
	progressive->format = DstFormat;
Packit 1fb8d4
Packit 1fb8d4
	while ((blocksLen - offset) >= 6)
Packit 1fb8d4
	{
Packit 1fb8d4
		PROGRESSIVE_BLOCK_SYNC sync;
Packit 1fb8d4
		PROGRESSIVE_BLOCK_CONTEXT context;
Packit 1fb8d4
		PROGRESSIVE_BLOCK_FRAME_BEGIN frameBegin;
Packit 1fb8d4
		PROGRESSIVE_BLOCK_FRAME_END frameEnd;
Packit 1fb8d4
		RFX_COMPONENT_CODEC_QUANT* quantVal;
Packit 1fb8d4
		RFX_PROGRESSIVE_CODEC_QUANT* quantProgVal;
Packit 1fb8d4
		boffset = 0;
Packit 1fb8d4
		block = &blocks[offset];
Packit 1fb8d4
		blockType = *((UINT16*) &block[boffset + 0]); /* blockType (2 bytes) */
Packit 1fb8d4
		blockLen = *((UINT32*) &block[boffset + 2]); /* blockLen (4 bytes) */
Packit 1fb8d4
		boffset += 6;
Packit 1fb8d4
Packit 1fb8d4
		if ((blocksLen - offset) < blockLen)
Packit 1fb8d4
			return -1003;
Packit 1fb8d4
Packit 1fb8d4
		switch (blockType)
Packit 1fb8d4
		{
Packit 1fb8d4
			case PROGRESSIVE_WBT_SYNC:
Packit 1fb8d4
				WLog_Print(progressive->log, WLOG_DEBUG, "ProgressiveSync");
Packit 1fb8d4
				sync.blockType = blockType;
Packit 1fb8d4
				sync.blockLen = blockLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) != 6)
Packit 1fb8d4
					return -1004;
Packit 1fb8d4
Packit 1fb8d4
				sync.magic = (UINT32) * ((UINT32*) &block[boffset + 0]); /* magic (4 bytes) */
Packit 1fb8d4
				sync.version = (UINT32) * ((UINT16*) &block[boffset +
Packit 1fb8d4
				                           4]); /* version (2 bytes) */
Packit 1fb8d4
				boffset += 6;
Packit 1fb8d4
Packit 1fb8d4
				if (sync.magic != 0xCACCACCA)
Packit 1fb8d4
					return -1005;
Packit 1fb8d4
Packit 1fb8d4
				if (sync.version != 0x0100)
Packit 1fb8d4
					return -1006;
Packit 1fb8d4
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_FRAME_BEGIN:
Packit 1fb8d4
				frameBegin.blockType = blockType;
Packit 1fb8d4
				frameBegin.blockLen = blockLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < 6)
Packit 1fb8d4
					return -1007;
Packit 1fb8d4
Packit 1fb8d4
				frameBegin.frameIndex = (UINT32) * ((UINT32*) &block[boffset +
Packit 1fb8d4
				                                    0]); /* frameIndex (4 bytes) */
Packit 1fb8d4
				frameBegin.regionCount = (UINT32) * ((UINT16*) &block[boffset +
Packit 1fb8d4
				                                     4]); /* regionCount (2 bytes) */
Packit 1fb8d4
				boffset += 6;
Packit 1fb8d4
				WLog_Print(progressive->log, WLOG_DEBUG,
Packit 1fb8d4
				           "ProgressiveFrameBegin: frameIndex: %"PRIu32" regionCount: %"PRIu16"",
Packit 1fb8d4
				           frameBegin.frameIndex, frameBegin.regionCount);
Packit 1fb8d4
				/**
Packit 1fb8d4
				 * If the number of elements specified by the regionCount field is
Packit 1fb8d4
				 * larger than the actual number of elements in the regions field,
Packit 1fb8d4
				 * the decoder SHOULD ignore this inconsistency.
Packit 1fb8d4
				 */
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_FRAME_END:
Packit 1fb8d4
				WLog_Print(progressive->log, WLOG_DEBUG, "ProgressiveFrameEnd");
Packit 1fb8d4
				frameEnd.blockType = blockType;
Packit 1fb8d4
				frameEnd.blockLen = blockLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) != 0)
Packit 1fb8d4
					return -1008;
Packit 1fb8d4
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_CONTEXT:
Packit 1fb8d4
				context.blockType = blockType;
Packit 1fb8d4
				context.blockLen = blockLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) != 4)
Packit 1fb8d4
					return -1009;
Packit 1fb8d4
Packit 1fb8d4
				context.ctxId = block[boffset + 0]; /* ctxId (1 byte) */
Packit 1fb8d4
				context.tileSize = *((UINT16*) &block[boffset + 1]); /* tileSize (2 bytes) */
Packit 1fb8d4
				context.flags = block[boffset + 3]; /* flags (1 byte) */
Packit 1fb8d4
				boffset += 4;
Packit 1fb8d4
Packit 1fb8d4
				if (context.tileSize != 64)
Packit 1fb8d4
					return -1010;
Packit 1fb8d4
Packit 1fb8d4
				WLog_Print(progressive->log, WLOG_DEBUG, "ProgressiveContext: flags: 0x%02"PRIX8"", context.flags);
Packit 1fb8d4
Packit 1fb8d4
				if (!(context.flags & RFX_SUBBAND_DIFFING))
Packit 1fb8d4
					WLog_Print(progressive->log, WLOG_WARN, "RFX_SUBBAND_DIFFING is not set");
Packit 1fb8d4
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			case PROGRESSIVE_WBT_REGION:
Packit 1fb8d4
				region->blockType = blockType;
Packit 1fb8d4
				region->blockLen = blockLen;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < 12)
Packit 1fb8d4
					return -1011;
Packit 1fb8d4
Packit 1fb8d4
				region->tileSize = block[boffset + 0]; /* tileSize (1 byte) */
Packit 1fb8d4
				region->numRects = *((UINT16*) &block[boffset + 1]); /* numRects (2 bytes) */
Packit 1fb8d4
				region->numQuant = block[boffset + 3]; /* numQuant (1 byte) */
Packit 1fb8d4
				region->numProgQuant = block[boffset + 4]; /* numProgQuant (1 byte) */
Packit 1fb8d4
				region->flags = block[boffset + 5]; /* flags (1 byte) */
Packit 1fb8d4
				region->numTiles = *((UINT16*) &block[boffset + 6]); /* numTiles (2 bytes) */
Packit 1fb8d4
				region->tileDataSize = *((UINT32*) &block[boffset +
Packit 1fb8d4
				                         8]); /* tileDataSize (4 bytes) */
Packit 1fb8d4
				boffset += 12;
Packit 1fb8d4
Packit 1fb8d4
				if (region->tileSize != 64)
Packit 1fb8d4
					return -1012;
Packit 1fb8d4
Packit 1fb8d4
				if (region->numRects < 1)
Packit 1fb8d4
					return -1013;
Packit 1fb8d4
Packit 1fb8d4
				if (region->numQuant > 7)
Packit 1fb8d4
					return -1014;
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < (region->numRects * 8))
Packit 1fb8d4
					return -1015;
Packit 1fb8d4
Packit 1fb8d4
				if (region->numRects > progressive->cRects)
Packit 1fb8d4
				{
Packit 1fb8d4
					RFX_RECT* tmpBuf = (RFX_RECT*) realloc(progressive->rects,
Packit 1fb8d4
					               region->numRects * sizeof(RFX_RECT));
Packit 1fb8d4
					if (!tmpBuf)
Packit 1fb8d4
						return -1016;
Packit 1fb8d4
Packit 1fb8d4
					progressive->rects = tmpBuf;
Packit 1fb8d4
					progressive->cRects = region->numRects;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				region->rects = progressive->rects;
Packit 1fb8d4
Packit 1fb8d4
				if (!region->rects)
Packit 1fb8d4
					return -1017;
Packit 1fb8d4
Packit 1fb8d4
				for (index = 0; index < region->numRects; index++)
Packit 1fb8d4
				{
Packit 1fb8d4
					rect = &(region->rects[index]);
Packit 1fb8d4
					rect->x = *((UINT16*) &block[boffset + 0]);
Packit 1fb8d4
					rect->y = *((UINT16*) &block[boffset + 2]);
Packit 1fb8d4
					rect->width = *((UINT16*) &block[boffset + 4]);
Packit 1fb8d4
					rect->height = *((UINT16*) &block[boffset + 6]);
Packit 1fb8d4
					boffset += 8;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < (region->numQuant * 5))
Packit 1fb8d4
					return -1018;
Packit 1fb8d4
Packit 1fb8d4
				if (region->numQuant > progressive->cQuant)
Packit 1fb8d4
				{
Packit 1fb8d4
					RFX_COMPONENT_CODEC_QUANT* tmpBuf = (RFX_COMPONENT_CODEC_QUANT*) realloc(
Packit 1fb8d4
					                             progressive->quantVals,
Packit 1fb8d4
					                             region->numQuant * sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
					if (!tmpBuf)
Packit 1fb8d4
						return -1019;
Packit 1fb8d4
Packit 1fb8d4
					progressive->quantVals = tmpBuf;
Packit 1fb8d4
					progressive->cQuant = region->numQuant;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				region->quantVals = progressive->quantVals;
Packit 1fb8d4
Packit 1fb8d4
				if (!region->quantVals)
Packit 1fb8d4
					return -1020;
Packit 1fb8d4
Packit 1fb8d4
				for (index = 0; index < region->numQuant; index++)
Packit 1fb8d4
				{
Packit 1fb8d4
					quantVal = &(region->quantVals[index]);
Packit 1fb8d4
					progressive_component_codec_quant_read(&block[boffset], quantVal);
Packit 1fb8d4
					boffset += 5;
Packit 1fb8d4
Packit 1fb8d4
					if (!progressive_rfx_quant_lcmp_greater_equal(quantVal, 6))
Packit 1fb8d4
						return -1;
Packit 1fb8d4
Packit 1fb8d4
					if (!progressive_rfx_quant_lcmp_less_equal(quantVal, 15))
Packit 1fb8d4
						return -1;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < (region->numProgQuant * 16))
Packit 1fb8d4
					return -1021;
Packit 1fb8d4
Packit 1fb8d4
				if (region->numProgQuant > progressive->cProgQuant)
Packit 1fb8d4
				{
Packit 1fb8d4
					RFX_PROGRESSIVE_CODEC_QUANT* tmpBuf = (RFX_PROGRESSIVE_CODEC_QUANT*) realloc(
Packit 1fb8d4
					                                 progressive->quantProgVals,
Packit 1fb8d4
					                                 region->numProgQuant * sizeof(RFX_PROGRESSIVE_CODEC_QUANT));
Packit 1fb8d4
					if (!tmpBuf)
Packit 1fb8d4
						return -1022;
Packit 1fb8d4
Packit 1fb8d4
					progressive->quantProgVals = tmpBuf;
Packit 1fb8d4
					progressive->cProgQuant = region->numProgQuant;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				region->quantProgVals = progressive->quantProgVals;
Packit 1fb8d4
Packit 1fb8d4
				if (!region->quantProgVals)
Packit 1fb8d4
					return -1023;
Packit 1fb8d4
Packit 1fb8d4
				for (index = 0; index < region->numProgQuant; index++)
Packit 1fb8d4
				{
Packit 1fb8d4
					quantProgVal = &(region->quantProgVals[index]);
Packit 1fb8d4
					quantProgVal->quality = block[boffset + 0];
Packit 1fb8d4
					progressive_component_codec_quant_read(&block[boffset + 1],
Packit 1fb8d4
					                                       &(quantProgVal->yQuantValues));
Packit 1fb8d4
					progressive_component_codec_quant_read(&block[boffset + 6],
Packit 1fb8d4
					                                       &(quantProgVal->cbQuantValues));
Packit 1fb8d4
					progressive_component_codec_quant_read(&block[boffset + 11],
Packit 1fb8d4
					                                       &(quantProgVal->crQuantValues));
Packit 1fb8d4
					boffset += 16;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				if ((blockLen - boffset) < region->tileDataSize)
Packit 1fb8d4
					return -1024;
Packit 1fb8d4
Packit 1fb8d4
				if (progressive->cTiles < surface->gridSize)
Packit 1fb8d4
				{
Packit 1fb8d4
					RFX_PROGRESSIVE_TILE** tmpBuf = (RFX_PROGRESSIVE_TILE**) realloc(progressive->tiles,
Packit 1fb8d4
					                     surface->gridSize * sizeof(RFX_PROGRESSIVE_TILE*));
Packit 1fb8d4
					if (!tmpBuf)
Packit 1fb8d4
						return -1025;
Packit 1fb8d4
Packit 1fb8d4
					progressive->tiles = tmpBuf;
Packit 1fb8d4
					progressive->cTiles = surface->gridSize;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				region->tiles = progressive->tiles;
Packit 1fb8d4
Packit 1fb8d4
				if (!region->tiles)
Packit 1fb8d4
					return -1;
Packit 1fb8d4
Packit 1fb8d4
				WLog_Print(progressive->log, WLOG_DEBUG,
Packit 1fb8d4
				           "ProgressiveRegion: numRects: %"PRIu16" numTiles: %"PRIu16" tileDataSize: %"PRIu32" flags: 0x%02"PRIX8" numQuant: %"PRIu8" numProgQuant: %"PRIu8"",
Packit 1fb8d4
				           region->numRects, region->numTiles, region->tileDataSize, region->flags,
Packit 1fb8d4
				           region->numQuant, region->numProgQuant);
Packit 1fb8d4
Packit 1fb8d4
				if (!(region->flags & RFX_DWT_REDUCE_EXTRAPOLATE))
Packit 1fb8d4
					WLog_Print(progressive->log, WLOG_WARN, "RFX_DWT_REDUCE_EXTRAPOLATE is not set");
Packit 1fb8d4
Packit 1fb8d4
				boxLeft = surface->gridWidth;
Packit 1fb8d4
				boxTop = surface->gridHeight;
Packit 1fb8d4
				boxRight = 0;
Packit 1fb8d4
				boxBottom = 0;
Packit 1fb8d4
Packit 1fb8d4
				for (index = 0; index < region->numRects; index++)
Packit 1fb8d4
				{
Packit 1fb8d4
					rect = &(region->rects[index]);
Packit 1fb8d4
					idxLeft = rect->x / 64;
Packit 1fb8d4
					idxTop = rect->y / 64;
Packit 1fb8d4
					idxRight = (rect->x + rect->width + 63) / 64;
Packit 1fb8d4
					idxBottom = (rect->y + rect->height + 63) / 64;
Packit 1fb8d4
Packit 1fb8d4
					if (idxLeft < boxLeft)
Packit 1fb8d4
						boxLeft = idxLeft;
Packit 1fb8d4
Packit 1fb8d4
					if (idxTop < boxTop)
Packit 1fb8d4
						boxTop = idxTop;
Packit 1fb8d4
Packit 1fb8d4
					if (idxRight > boxRight)
Packit 1fb8d4
						boxRight = idxRight;
Packit 1fb8d4
Packit 1fb8d4
					if (idxBottom > boxBottom)
Packit 1fb8d4
						boxBottom = idxBottom;
Packit 1fb8d4
Packit 1fb8d4
					WLog_Print(progressive->log, WLOG_DEBUG,
Packit 1fb8d4
					           "rect[%"PRIu16"]: x: %"PRIu16" y: %"PRIu16" w: %"PRIu16" h: %"PRIu16"",
Packit 1fb8d4
					           index, rect->x, rect->y, rect->width, rect->height);
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				status = progressive_process_tiles(progressive, &block[boffset],
Packit 1fb8d4
				                                   region->tileDataSize, surface);
Packit 1fb8d4
Packit 1fb8d4
				if (status < 0)
Packit 1fb8d4
					return status;
Packit 1fb8d4
Packit 1fb8d4
				region->numTiles = 0;
Packit 1fb8d4
Packit 1fb8d4
				for (index = 0; index < surface->gridSize; index++)
Packit 1fb8d4
				{
Packit 1fb8d4
					RFX_PROGRESSIVE_TILE* tile = &(surface->tiles[index]);
Packit 1fb8d4
Packit 1fb8d4
					if (!tile->data)
Packit 1fb8d4
						continue;
Packit 1fb8d4
Packit 1fb8d4
					if ((tile->xIdx < boxLeft) || (tile->xIdx > boxRight))
Packit 1fb8d4
						continue;
Packit 1fb8d4
Packit 1fb8d4
					if ((tile->yIdx < boxTop) || (tile->yIdx > boxBottom))
Packit 1fb8d4
						continue;
Packit 1fb8d4
Packit 1fb8d4
					region->tiles[region->numTiles++] = tile;
Packit 1fb8d4
				}
Packit 1fb8d4
Packit 1fb8d4
				boffset += (UINT32) status;
Packit 1fb8d4
				break;
Packit 1fb8d4
Packit 1fb8d4
			default:
Packit 1fb8d4
				return -1039;
Packit 1fb8d4
				break;
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		if (boffset != blockLen)
Packit 1fb8d4
			return -1040;
Packit 1fb8d4
Packit 1fb8d4
		offset += blockLen;
Packit 1fb8d4
		count++;
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	if (offset != blocksLen)
Packit 1fb8d4
		return -1041;
Packit 1fb8d4
Packit 1fb8d4
	region = &(progressive->region);
Packit 1fb8d4
	region16_init(&clippingRects);
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < region->numRects; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		RECTANGLE_16 clippingRect;
Packit 1fb8d4
		const RFX_RECT* rect = &(region->rects[i]);
Packit 1fb8d4
		clippingRect.left = nXDst + rect->x;
Packit 1fb8d4
		clippingRect.top = nYDst + rect->y;
Packit 1fb8d4
		clippingRect.right = clippingRect.left + rect->width;
Packit 1fb8d4
		clippingRect.bottom = clippingRect.top + rect->height;
Packit 1fb8d4
		region16_union_rect(&clippingRects, &clippingRects, &clippingRect);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	for (i = 0; i < region->numTiles; i++)
Packit 1fb8d4
	{
Packit 1fb8d4
		UINT32 nbUpdateRects;
Packit 1fb8d4
		const RECTANGLE_16* updateRects;
Packit 1fb8d4
		RECTANGLE_16 updateRect;
Packit 1fb8d4
		RFX_PROGRESSIVE_TILE* tile = region->tiles[i];
Packit 1fb8d4
		updateRect.left = nXDst + tile->x;
Packit 1fb8d4
		updateRect.top = nYDst + tile->y;
Packit 1fb8d4
		updateRect.right = updateRect.left + 64;
Packit 1fb8d4
		updateRect.bottom = updateRect.top + 64;
Packit 1fb8d4
		region16_init(&updateRegion);
Packit 1fb8d4
		region16_intersect_rect(&updateRegion, &clippingRects, &updateRect);
Packit 1fb8d4
		updateRects = region16_rects(&updateRegion, &nbUpdateRects);
Packit 1fb8d4
Packit 1fb8d4
		for (j = 0; j < nbUpdateRects; j++)
Packit 1fb8d4
		{
Packit 1fb8d4
			const RECTANGLE_16* rect = &updateRects[j];
Packit 1fb8d4
			const UINT32 nXSrc = rect->left - (nXDst + tile->x);
Packit 1fb8d4
			const UINT32 nYSrc = rect->top - (nYDst + tile->y);
Packit 1fb8d4
			const UINT32 width = rect->right - rect->left;
Packit 1fb8d4
			const UINT32 height = rect->bottom - rect->top;
Packit 1fb8d4
Packit 1fb8d4
			if (!freerdp_image_copy(pDstData, DstFormat, nDstStep,
Packit 1fb8d4
			                        rect->left, rect->top,
Packit 1fb8d4
			                        width, height, tile->data, tile->format,
Packit 1fb8d4
			                        tile->stride,
Packit 1fb8d4
			                        nXSrc, nYSrc, NULL, FREERDP_FLIP_NONE))
Packit 1fb8d4
			{
Packit 1fb8d4
				rc = -42;
Packit 1fb8d4
				break;
Packit 1fb8d4
			}
Packit 1fb8d4
Packit 1fb8d4
			if (invalidRegion)
Packit 1fb8d4
				region16_union_rect(invalidRegion, invalidRegion, rect);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		region16_uninit(&updateRegion);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	region16_uninit(&clippingRects);
Packit 1fb8d4
	return rc;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
int progressive_compress(PROGRESSIVE_CONTEXT* progressive, const BYTE* pSrcData,
Packit 1fb8d4
                         UINT32 SrcSize, BYTE** ppDstData, UINT32* pDstSize)
Packit 1fb8d4
{
Packit 1fb8d4
	return -1;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
BOOL progressive_context_reset(PROGRESSIVE_CONTEXT* progressive)
Packit 1fb8d4
{
Packit 1fb8d4
	if (!progressive)
Packit 1fb8d4
		return FALSE;
Packit 1fb8d4
Packit 1fb8d4
	return TRUE;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
PROGRESSIVE_CONTEXT* progressive_context_new(BOOL Compressor)
Packit 1fb8d4
{
Packit 1fb8d4
	PROGRESSIVE_CONTEXT* progressive;
Packit 1fb8d4
	progressive = (PROGRESSIVE_CONTEXT*) calloc(1, sizeof(PROGRESSIVE_CONTEXT));
Packit 1fb8d4
Packit 1fb8d4
	if (progressive)
Packit 1fb8d4
	{
Packit 1fb8d4
		progressive->Compressor = Compressor;
Packit 1fb8d4
		progressive->bufferPool = BufferPool_New(TRUE, (8192 + 32) * 3, 16);
Packit 1fb8d4
		progressive->cRects = 64;
Packit 1fb8d4
		progressive->rects = (RFX_RECT*) calloc(progressive->cRects, sizeof(RFX_RECT));
Packit 1fb8d4
Packit 1fb8d4
		if (!progressive->rects)
Packit 1fb8d4
			goto cleanup;
Packit 1fb8d4
Packit 1fb8d4
		progressive->cTiles = 64;
Packit 1fb8d4
		progressive->tiles = (RFX_PROGRESSIVE_TILE**) calloc(progressive->cTiles,
Packit 1fb8d4
		                     sizeof(RFX_PROGRESSIVE_TILE*));
Packit 1fb8d4
Packit 1fb8d4
		if (!progressive->tiles)
Packit 1fb8d4
			goto cleanup;
Packit 1fb8d4
Packit 1fb8d4
		progressive->cQuant = 8;
Packit 1fb8d4
		progressive->quantVals = (RFX_COMPONENT_CODEC_QUANT*) calloc(
Packit 1fb8d4
		                             progressive->cQuant, sizeof(RFX_COMPONENT_CODEC_QUANT));
Packit 1fb8d4
Packit 1fb8d4
		if (!progressive->quantVals)
Packit 1fb8d4
			goto cleanup;
Packit 1fb8d4
Packit 1fb8d4
		progressive->cProgQuant = 8;
Packit 1fb8d4
		progressive->quantProgVals = (RFX_PROGRESSIVE_CODEC_QUANT*) calloc(
Packit 1fb8d4
		                                 progressive->cProgQuant, sizeof(RFX_PROGRESSIVE_CODEC_QUANT));
Packit 1fb8d4
Packit 1fb8d4
		if (!progressive->quantProgVals)
Packit 1fb8d4
			goto cleanup;
Packit 1fb8d4
Packit 1fb8d4
		ZeroMemory(&(progressive->quantProgValFull),
Packit 1fb8d4
		           sizeof(RFX_PROGRESSIVE_CODEC_QUANT));
Packit 1fb8d4
		progressive->quantProgValFull.quality = 100;
Packit 1fb8d4
		progressive->SurfaceContexts = HashTable_New(TRUE);
Packit 1fb8d4
		progressive_context_reset(progressive);
Packit 1fb8d4
		progressive->log = WLog_Get(TAG);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	return progressive;
Packit 1fb8d4
cleanup:
Packit 1fb8d4
	progressive_context_free(progressive);
Packit 1fb8d4
	return NULL;
Packit 1fb8d4
}
Packit 1fb8d4
Packit 1fb8d4
void progressive_context_free(PROGRESSIVE_CONTEXT* progressive)
Packit 1fb8d4
{
Packit 1fb8d4
	int count;
Packit 1fb8d4
	int index;
Packit 1fb8d4
	ULONG_PTR* pKeys = NULL;
Packit 1fb8d4
	PROGRESSIVE_SURFACE_CONTEXT* surface;
Packit 1fb8d4
Packit 1fb8d4
	if (!progressive)
Packit 1fb8d4
		return;
Packit 1fb8d4
Packit 1fb8d4
	BufferPool_Free(progressive->bufferPool);
Packit 1fb8d4
	free(progressive->rects);
Packit 1fb8d4
	free(progressive->tiles);
Packit 1fb8d4
	free(progressive->quantVals);
Packit 1fb8d4
	free(progressive->quantProgVals);
Packit 1fb8d4
Packit 1fb8d4
	if (progressive->SurfaceContexts)
Packit 1fb8d4
	{
Packit 1fb8d4
		count = HashTable_GetKeys(progressive->SurfaceContexts, &pKeys);
Packit 1fb8d4
Packit 1fb8d4
		for (index = 0; index < count; index++)
Packit 1fb8d4
		{
Packit 1fb8d4
			surface = (PROGRESSIVE_SURFACE_CONTEXT*) HashTable_GetItemValue(progressive->SurfaceContexts, (void*) pKeys[index]);
Packit 1fb8d4
			progressive_surface_context_free(surface);
Packit 1fb8d4
		}
Packit 1fb8d4
Packit 1fb8d4
		free(pKeys);
Packit 1fb8d4
		HashTable_Free(progressive->SurfaceContexts);
Packit 1fb8d4
	}
Packit 1fb8d4
Packit 1fb8d4
	free(progressive);
Packit 1fb8d4
}
Packit 1fb8d4