|
|
@ -72,19 +72,22 @@ typedef struct TsccContext { |
|
|
|
* |
|
|
|
* |
|
|
|
*/ |
|
|
|
*/ |
|
|
|
|
|
|
|
|
|
|
|
static int decode_rle(CamtasiaContext *c) |
|
|
|
static int decode_rle(CamtasiaContext *c, unsigned int srcsize) |
|
|
|
{ |
|
|
|
{ |
|
|
|
unsigned char *src = c->decomp_buf; |
|
|
|
unsigned char *src = c->decomp_buf; |
|
|
|
unsigned char *output; |
|
|
|
unsigned char *output, *output_end; |
|
|
|
int p1, p2, line=c->height, pos=0, i; |
|
|
|
int p1, p2, line=c->height, pos=0, i; |
|
|
|
|
|
|
|
|
|
|
|
output = c->pic.data[0] + (c->height - 1) * c->pic.linesize[0]; |
|
|
|
output = c->pic.data[0] + (c->height - 1) * c->pic.linesize[0]; |
|
|
|
while(src < c->decomp_buf + c->decomp_size) { |
|
|
|
output_end = c->pic.data[0] + (c->height) * c->pic.linesize[0]; |
|
|
|
|
|
|
|
while(src < c->decomp_buf + srcsize) { |
|
|
|
p1 = *src++; |
|
|
|
p1 = *src++; |
|
|
|
if(p1 == 0) { //Escape code
|
|
|
|
if(p1 == 0) { //Escape code
|
|
|
|
p2 = *src++; |
|
|
|
p2 = *src++; |
|
|
|
if(p2 == 0) { //End-of-line
|
|
|
|
if(p2 == 0) { //End-of-line
|
|
|
|
output = c->pic.data[0] + (--line) * c->pic.linesize[0]; |
|
|
|
output = c->pic.data[0] + (--line) * c->pic.linesize[0]; |
|
|
|
|
|
|
|
if (line < 0) |
|
|
|
|
|
|
|
return -1; |
|
|
|
pos = 0; |
|
|
|
pos = 0; |
|
|
|
continue; |
|
|
|
continue; |
|
|
|
} else if(p2 == 1) { //End-of-picture
|
|
|
|
} else if(p2 == 1) { //End-of-picture
|
|
|
@ -93,11 +96,17 @@ static int decode_rle(CamtasiaContext *c) |
|
|
|
p1 = *src++; |
|
|
|
p1 = *src++; |
|
|
|
p2 = *src++; |
|
|
|
p2 = *src++; |
|
|
|
line -= p2; |
|
|
|
line -= p2; |
|
|
|
|
|
|
|
if (line < 0) |
|
|
|
|
|
|
|
return -1; |
|
|
|
pos += p1; |
|
|
|
pos += p1; |
|
|
|
output = c->pic.data[0] + line * c->pic.linesize[0] + pos * (c->bpp / 8); |
|
|
|
output = c->pic.data[0] + line * c->pic.linesize[0] + pos * (c->bpp / 8); |
|
|
|
continue; |
|
|
|
continue; |
|
|
|
} |
|
|
|
} |
|
|
|
// Copy data
|
|
|
|
// Copy data
|
|
|
|
|
|
|
|
if (output + p2 * (c->bpp / 8) > output_end) { |
|
|
|
|
|
|
|
src += p2 * (c->bpp / 8); |
|
|
|
|
|
|
|
continue; |
|
|
|
|
|
|
|
} |
|
|
|
for(i = 0; i < p2 * (c->bpp / 8); i++) { |
|
|
|
for(i = 0; i < p2 * (c->bpp / 8); i++) { |
|
|
|
*output++ = *src++; |
|
|
|
*output++ = *src++; |
|
|
|
} |
|
|
|
} |
|
|
@ -119,6 +128,8 @@ static int decode_rle(CamtasiaContext *c) |
|
|
|
pix[2] = *src++; |
|
|
|
pix[2] = *src++; |
|
|
|
break; |
|
|
|
break; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
if (output + p1 * (c->bpp / 8) > output_end) |
|
|
|
|
|
|
|
continue; |
|
|
|
for(i = 0; i < p1; i++) { |
|
|
|
for(i = 0; i < p1; i++) { |
|
|
|
switch(c->bpp){ |
|
|
|
switch(c->bpp){ |
|
|
|
case 8: *output++ = pix[0]; |
|
|
|
case 8: *output++ = pix[0]; |
|
|
@ -183,10 +194,10 @@ static int decode_frame(AVCodecContext *avctx, void *data, int *data_size, uint8 |
|
|
|
av_log(avctx, AV_LOG_ERROR, "Inflate error: %d\n", zret); |
|
|
|
av_log(avctx, AV_LOG_ERROR, "Inflate error: %d\n", zret); |
|
|
|
return -1; |
|
|
|
return -1; |
|
|
|
} |
|
|
|
} |
|
|
|
encoded = c->decomp_buf; |
|
|
|
|
|
|
|
len = c->decomp_size; |
|
|
|
|
|
|
|
if(zret != Z_DATA_ERROR) |
|
|
|
if(zret != Z_DATA_ERROR) |
|
|
|
decode_rle(c); |
|
|
|
decode_rle(c, c->zstream.avail_out); |
|
|
|
|
|
|
|
|
|
|
|
/* make the palette available on the way out */ |
|
|
|
/* make the palette available on the way out */ |
|
|
|
if (c->avctx->pix_fmt == PIX_FMT_PAL8) { |
|
|
|
if (c->avctx->pix_fmt == PIX_FMT_PAL8) { |
|
|
@ -227,6 +238,10 @@ static int decode_init(AVCodecContext *avctx) |
|
|
|
c->pic.data[0] = NULL; |
|
|
|
c->pic.data[0] = NULL; |
|
|
|
c->height = avctx->height; |
|
|
|
c->height = avctx->height; |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (avcodec_check_dimensions(avctx, avctx->height, avctx->width) < 0) { |
|
|
|
|
|
|
|
return 1; |
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
|
|
#ifdef CONFIG_ZLIB |
|
|
|
#ifdef CONFIG_ZLIB |
|
|
|
// Needed if zlib unused or init aborted before inflateInit
|
|
|
|
// Needed if zlib unused or init aborted before inflateInit
|
|
|
|
memset(&(c->zstream), 0, sizeof(z_stream));
|
|
|
|
memset(&(c->zstream), 0, sizeof(z_stream));
|
|
|
|