|  |  | @@ -149,6 +149,80 @@ static int zmbv_decode_xor_8(ZmbvContext *c) | 
		
	
		
			
			|  |  |  | return 0; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | /** | 
		
	
		
			
			|  |  |  | * Decode XOR'ed frame - 15bpp and 16bpp version | 
		
	
		
			
			|  |  |  | */ | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | static int zmbv_decode_xor_16(ZmbvContext *c) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | uint8_t *src = c->decomp_buf; | 
		
	
		
			
			|  |  |  | uint16_t *output, *prev; | 
		
	
		
			
			|  |  |  | int8_t *mvec; | 
		
	
		
			
			|  |  |  | int x, y; | 
		
	
		
			
			|  |  |  | int d, dx, dy, bw2, bh2; | 
		
	
		
			
			|  |  |  | int block; | 
		
	
		
			
			|  |  |  | int i, j; | 
		
	
		
			
			|  |  |  | int mx, my; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | output = (uint16_t*)c->cur; | 
		
	
		
			
			|  |  |  | prev = (uint16_t*)c->prev; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | mvec = (int8_t*)src; | 
		
	
		
			
			|  |  |  | src += ((c->bx * c->by * 2 + 3) & ~3); | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | block = 0; | 
		
	
		
			
			|  |  |  | for(y = 0; y < c->height; y += c->bh) { | 
		
	
		
			
			|  |  |  | bh2 = ((c->height - y) > c->bh) ? c->bh : (c->height - y); | 
		
	
		
			
			|  |  |  | for(x = 0; x < c->width; x += c->bw) { | 
		
	
		
			
			|  |  |  | uint16_t *out, *tprev; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | d = mvec[block] & 1; | 
		
	
		
			
			|  |  |  | dx = mvec[block] >> 1; | 
		
	
		
			
			|  |  |  | dy = mvec[block + 1] >> 1; | 
		
	
		
			
			|  |  |  | block += 2; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | bw2 = ((c->width - x) > c->bw) ? c->bw : (c->width - x); | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | /* copy block - motion vectors out of bounds are used to zero blocks */ | 
		
	
		
			
			|  |  |  | out = output + x; | 
		
	
		
			
			|  |  |  | tprev = prev + x + dx + dy * c->width; | 
		
	
		
			
			|  |  |  | mx = x + dx; | 
		
	
		
			
			|  |  |  | my = y + dy; | 
		
	
		
			
			|  |  |  | for(j = 0; j < bh2; j++){ | 
		
	
		
			
			|  |  |  | if((my + j < 0) || (my + j >= c->height)) { | 
		
	
		
			
			|  |  |  | memset(out, 0, bw2 * 2); | 
		
	
		
			
			|  |  |  | } else { | 
		
	
		
			
			|  |  |  | for(i = 0; i < bw2; i++){ | 
		
	
		
			
			|  |  |  | if((mx + i < 0) || (mx + i >= c->width)) | 
		
	
		
			
			|  |  |  | out[i] = 0; | 
		
	
		
			
			|  |  |  | else | 
		
	
		
			
			|  |  |  | out[i] = tprev[i]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->width; | 
		
	
		
			
			|  |  |  | tprev += c->width; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | if(d) { /* apply XOR'ed difference */ | 
		
	
		
			
			|  |  |  | out = output + x; | 
		
	
		
			
			|  |  |  | for(j = 0; j < bh2; j++){ | 
		
	
		
			
			|  |  |  | for(i = 0; i < bw2; i++) { | 
		
	
		
			
			|  |  |  | out[i] ^= *((uint16_t*)src); | 
		
	
		
			
			|  |  |  | src += 2; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->width; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | output += c->width * c->bh; | 
		
	
		
			
			|  |  |  | prev += c->width * c->bh; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | if(src - c->decomp_buf != c->decomp_len) | 
		
	
		
			
			|  |  |  | av_log(c->avctx, AV_LOG_ERROR, "Used %i of %i bytes\n", src-c->decomp_buf, c->decomp_len); | 
		
	
		
			
			|  |  |  | return 0; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | #ifdef ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | /** | 
		
	
		
			
			|  |  |  | * Decode XOR'ed frame - 24bpp version | 
		
	
		
			
			|  |  |  | */ | 
		
	
	
		
			
				|  |  | @@ -229,6 +303,80 @@ static int zmbv_decode_xor_24(ZmbvContext *c) | 
		
	
		
			
			|  |  |  | av_log(c->avctx, AV_LOG_ERROR, "Used %i of %i bytes\n", src-c->decomp_buf, c->decomp_len); | 
		
	
		
			
			|  |  |  | return 0; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | #endif //ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | /** | 
		
	
		
			
			|  |  |  | * Decode XOR'ed frame - 32bpp version | 
		
	
		
			
			|  |  |  | */ | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | static int zmbv_decode_xor_32(ZmbvContext *c) | 
		
	
		
			
			|  |  |  | { | 
		
	
		
			
			|  |  |  | uint8_t *src = c->decomp_buf; | 
		
	
		
			
			|  |  |  | uint32_t *output, *prev; | 
		
	
		
			
			|  |  |  | int8_t *mvec; | 
		
	
		
			
			|  |  |  | int x, y; | 
		
	
		
			
			|  |  |  | int d, dx, dy, bw2, bh2; | 
		
	
		
			
			|  |  |  | int block; | 
		
	
		
			
			|  |  |  | int i, j; | 
		
	
		
			
			|  |  |  | int mx, my; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | output = (uint32_t*)c->cur; | 
		
	
		
			
			|  |  |  | prev = (uint32_t*)c->prev; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | mvec = (int8_t*)src; | 
		
	
		
			
			|  |  |  | src += ((c->bx * c->by * 2 + 3) & ~3); | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | block = 0; | 
		
	
		
			
			|  |  |  | for(y = 0; y < c->height; y += c->bh) { | 
		
	
		
			
			|  |  |  | bh2 = ((c->height - y) > c->bh) ? c->bh : (c->height - y); | 
		
	
		
			
			|  |  |  | for(x = 0; x < c->width; x += c->bw) { | 
		
	
		
			
			|  |  |  | uint32_t *out, *tprev; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | d = mvec[block] & 1; | 
		
	
		
			
			|  |  |  | dx = mvec[block] >> 1; | 
		
	
		
			
			|  |  |  | dy = mvec[block + 1] >> 1; | 
		
	
		
			
			|  |  |  | block += 2; | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | bw2 = ((c->width - x) > c->bw) ? c->bw : (c->width - x); | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | /* copy block - motion vectors out of bounds are used to zero blocks */ | 
		
	
		
			
			|  |  |  | out = output + x; | 
		
	
		
			
			|  |  |  | tprev = prev + x + dx + dy * c->width; | 
		
	
		
			
			|  |  |  | mx = x + dx; | 
		
	
		
			
			|  |  |  | my = y + dy; | 
		
	
		
			
			|  |  |  | for(j = 0; j < bh2; j++){ | 
		
	
		
			
			|  |  |  | if((my + j < 0) || (my + j >= c->height)) { | 
		
	
		
			
			|  |  |  | memset(out, 0, bw2 * 4); | 
		
	
		
			
			|  |  |  | } else { | 
		
	
		
			
			|  |  |  | for(i = 0; i < bw2; i++){ | 
		
	
		
			
			|  |  |  | if((mx + i < 0) || (mx + i >= c->width)) | 
		
	
		
			
			|  |  |  | out[i] = 0; | 
		
	
		
			
			|  |  |  | else | 
		
	
		
			
			|  |  |  | out[i] = tprev[i]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->width; | 
		
	
		
			
			|  |  |  | tprev += c->width; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | if(d) { /* apply XOR'ed difference */ | 
		
	
		
			
			|  |  |  | out = output + x; | 
		
	
		
			
			|  |  |  | for(j = 0; j < bh2; j++){ | 
		
	
		
			
			|  |  |  | for(i = 0; i < bw2; i++) { | 
		
	
		
			
			|  |  |  | out[i] ^= *((uint32_t*)src); | 
		
	
		
			
			|  |  |  | src += 4; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->width; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | output += c->width * c->bh; | 
		
	
		
			
			|  |  |  | prev += c->width * c->bh; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | if(src - c->decomp_buf != c->decomp_len) | 
		
	
		
			
			|  |  |  | av_log(c->avctx, AV_LOG_ERROR, "Used %i of %i bytes\n", src-c->decomp_buf, c->decomp_len); | 
		
	
		
			
			|  |  |  | return 0; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | /** | 
		
	
		
			
			|  |  |  | * Decode intraframe | 
		
	
	
		
			
				|  |  | @@ -294,7 +442,34 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, uint8 | 
		
	
		
			
			|  |  |  | av_log(avctx, AV_LOG_ERROR, "Unsupported compression type %i\n", c->comp); | 
		
	
		
			
			|  |  |  | return -1; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | if(c->fmt != ZMBV_FMT_8BPP && c->fmt != ZMBV_FMT_24BPP) { | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | switch(c->fmt) { | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_8BPP: | 
		
	
		
			
			|  |  |  | c->bpp = 8; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_8; | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_15BPP: | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_16BPP: | 
		
	
		
			
			|  |  |  | c->bpp = 16; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_16; | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | #ifdef ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_24BPP: | 
		
	
		
			
			|  |  |  | c->bpp = 24; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_24; | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | #endif //ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_32BPP: | 
		
	
		
			
			|  |  |  | c->bpp = 32; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_32; | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | default: | 
		
	
		
			
			|  |  |  | c->decode_intra = NULL; | 
		
	
		
			
			|  |  |  | c->decode_xor = NULL; | 
		
	
		
			
			|  |  |  | av_log(avctx, AV_LOG_ERROR, "Unsupported (for now) format %i\n", c->fmt); | 
		
	
		
			
			|  |  |  | return -1; | 
		
	
		
			
			|  |  |  | } | 
		
	
	
		
			
				|  |  | @@ -308,22 +483,13 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, uint8 | 
		
	
		
			
			|  |  |  | av_log(avctx, AV_LOG_ERROR, "BUG! Zlib support not compiled in frame decoder.\n"); | 
		
	
		
			
			|  |  |  | return -1; | 
		
	
		
			
			|  |  |  | #endif  /* CONFIG_ZLIB */ | 
		
	
		
			
			|  |  |  | if(c->fmt == ZMBV_FMT_8BPP) { | 
		
	
		
			
			|  |  |  | c->bpp = 8; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_8; | 
		
	
		
			
			|  |  |  | } else { | 
		
	
		
			
			|  |  |  | c->bpp = 24; | 
		
	
		
			
			|  |  |  | c->decode_intra = zmbv_decode_intra; | 
		
	
		
			
			|  |  |  | c->decode_xor = zmbv_decode_xor_24; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | c->cur = av_realloc(c->cur, avctx->width * avctx->height * (c->bpp / 8)); | 
		
	
		
			
			|  |  |  | c->prev = av_realloc(c->prev, avctx->width * avctx->height * (c->bpp / 8)); | 
		
	
		
			
			|  |  |  | c->bx = (c->width + c->bw - 1) / c->bw; | 
		
	
		
			
			|  |  |  | c->by = (c->height+ c->bh - 1) / c->bh; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | if(c->fmt == 0) { | 
		
	
		
			
			|  |  |  | if(c->decode_intra == NULL) { | 
		
	
		
			
			|  |  |  | av_log(avctx, AV_LOG_ERROR, "Error! Got no format or no keyframe!\n"); | 
		
	
		
			
			|  |  |  | return -1; | 
		
	
		
			
			|  |  |  | } | 
		
	
	
		
			
				|  |  | @@ -362,16 +528,67 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, uint8 | 
		
	
		
			
			|  |  |  | 
 | 
		
	
		
			
			|  |  |  | out = c->pic.data[0]; | 
		
	
		
			
			|  |  |  | src = c->cur; | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | for(i = 0; i < c->width; i++) { | 
		
	
		
			
			|  |  |  | out[i * 3 + 0] = c->pal[(*src) * 3 + 0]; | 
		
	
		
			
			|  |  |  | out[i * 3 + 1] = c->pal[(*src) * 3 + 1]; | 
		
	
		
			
			|  |  |  | out[i * 3 + 2] = c->pal[(*src) * 3 + 2]; | 
		
	
		
			
			|  |  |  | *src++; | 
		
	
		
			
			|  |  |  | switch(c->fmt) { | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_8BPP: | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | for(i = 0; i < c->width; i++) { | 
		
	
		
			
			|  |  |  | out[i * 3 + 0] = c->pal[(*src) * 3 + 0]; | 
		
	
		
			
			|  |  |  | out[i * 3 + 1] = c->pal[(*src) * 3 + 1]; | 
		
	
		
			
			|  |  |  | out[i * 3 + 2] = c->pal[(*src) * 3 + 2]; | 
		
	
		
			
			|  |  |  | *src++; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_15BPP: | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | for(i = 0; i < c->width; i++) { | 
		
	
		
			
			|  |  |  | uint16_t tmp = LE_16(src); | 
		
	
		
			
			|  |  |  | src += 2; | 
		
	
		
			
			|  |  |  | out[i * 3 + 0] = (tmp & 0x7C00) >> 7; | 
		
	
		
			
			|  |  |  | out[i * 3 + 1] = (tmp & 0x03E0) >> 2; | 
		
	
		
			
			|  |  |  | out[i * 3 + 2] = (tmp & 0x001F) << 3; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_16BPP: | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | for(i = 0; i < c->width; i++) { | 
		
	
		
			
			|  |  |  | uint16_t tmp = LE_16(src); | 
		
	
		
			
			|  |  |  | src += 2; | 
		
	
		
			
			|  |  |  | out[i * 3 + 0] = (tmp & 0xF800) >> 8; | 
		
	
		
			
			|  |  |  | out[i * 3 + 1] = (tmp & 0x07E0) >> 3; | 
		
	
		
			
			|  |  |  | out[i * 3 + 2] = (tmp & 0x001F) << 3; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | #ifdef ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_24BPP: | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | memcpy(out, src, c->width * 3); | 
		
	
		
			
			|  |  |  | src += c->width * 3; | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | #endif //ZMBV_ENABLE_24BPP | 
		
	
		
			
			|  |  |  | case ZMBV_FMT_32BPP: | 
		
	
		
			
			|  |  |  | for(j = 0; j < c->height; j++) { | 
		
	
		
			
			|  |  |  | for(i = 0; i < c->width; i++) { | 
		
	
		
			
			|  |  |  | uint32_t tmp = LE_32(src); | 
		
	
		
			
			|  |  |  | src += 4; | 
		
	
		
			
			|  |  |  | out[i * 3 + 0] = tmp >> 16; | 
		
	
		
			
			|  |  |  | out[i * 3 + 1] = tmp >> 8; | 
		
	
		
			
			|  |  |  | out[i * 3 + 2] = tmp >> 0; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | out += c->pic.linesize[0]; | 
		
	
		
			
			|  |  |  | break; | 
		
	
		
			
			|  |  |  | default: | 
		
	
		
			
			|  |  |  | av_log(avctx, AV_LOG_ERROR, "Cannot handle format %i\n", c->fmt); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | memcpy(c->prev, c->cur, c->width * c->height); | 
		
	
		
			
			|  |  |  | memcpy(c->prev, c->cur, c->width * c->height * (c->bpp / 8)); | 
		
	
		
			
			|  |  |  | } | 
		
	
		
			
			|  |  |  | *data_size = sizeof(AVFrame); | 
		
	
		
			
			|  |  |  | *(AVFrame*)data = c->pic; | 
		
	
	
		
			
				|  |  | 
 |