mirror of
https://github.com/mpv-player/mpv
synced 2025-01-20 21:07:29 +01:00
- fixed some bugs in RLE decoder
- some cleanup, home i didn't break it... git-svn-id: svn://svn.mplayerhq.hu/mplayer/trunk@7402 b3059339-0415-0410-9bf9-f77b7e298cf2
This commit is contained in:
parent
977bc3ded4
commit
1adce2c92a
@ -20,7 +20,7 @@ static vd_info_t info =
|
|||||||
{
|
{
|
||||||
"TGA Images decoder",
|
"TGA Images decoder",
|
||||||
"mtga",
|
"mtga",
|
||||||
"Tilman Sauerbeck",
|
"Tilman Sauerbeck, A'rpi",
|
||||||
"Tilman Sauerbeck",
|
"Tilman Sauerbeck",
|
||||||
"only 24bpp and 32bpp RGB targa files support so far"
|
"only 24bpp and 32bpp RGB targa files support so far"
|
||||||
};
|
};
|
||||||
@ -63,20 +63,13 @@ static int last_c = -1;
|
|||||||
static int control(sh_video_t *sh, int cmd, void *arg, ...)
|
static int control(sh_video_t *sh, int cmd, void *arg, ...)
|
||||||
{
|
{
|
||||||
TGAInfo *info = (TGAInfo *) sh->context;
|
TGAInfo *info = (TGAInfo *) sh->context;
|
||||||
|
|
||||||
|
|
||||||
switch (cmd)
|
switch (cmd)
|
||||||
{
|
{
|
||||||
case VDCTRL_QUERY_FORMAT:
|
case VDCTRL_QUERY_FORMAT:
|
||||||
if ((*((int *) arg) == IMGFMT_BGR32 && info->bpp == 32) || (*((int *) arg) == IMGFMT_BGR24 && info->bpp == 24))
|
if (*((int *) arg) == out_fmt) return CONTROL_TRUE;
|
||||||
return CONTROL_TRUE;
|
return CONTROL_FALSE;
|
||||||
else
|
|
||||||
return CONTROL_FALSE;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
return CONTROL_UNKNOWN;
|
|
||||||
break;
|
|
||||||
}
|
}
|
||||||
|
return CONTROL_UNKNOWN;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* init driver */
|
/* init driver */
|
||||||
@ -85,7 +78,6 @@ static int init(sh_video_t *sh)
|
|||||||
sh->context = (TGAInfo *) calloc(1, sizeof(TGAInfo));
|
sh->context = (TGAInfo *) calloc(1, sizeof(TGAInfo));
|
||||||
last_w = -1;
|
last_w = -1;
|
||||||
|
|
||||||
|
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -94,89 +86,85 @@ static int init(sh_video_t *sh)
|
|||||||
static void uninit(sh_video_t *sh)
|
static void uninit(sh_video_t *sh)
|
||||||
{
|
{
|
||||||
TGAInfo *info = sh->context;
|
TGAInfo *info = sh->context;
|
||||||
|
|
||||||
|
|
||||||
free(info);
|
free(info);
|
||||||
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* decode a runlength-encoded tga */
|
/* decode a runlength-encoded tga */
|
||||||
static void decode_rle_tga(TGAInfo *info, unsigned char *data, mp_image_t **mpi)
|
static void decode_rle_tga(TGAInfo *info, unsigned char *data, mp_image_t *mpi)
|
||||||
{
|
{
|
||||||
short i, num_bytes = info->bpp / 8;
|
int row, col, replen, i, num_bytes = info->bpp / 8;
|
||||||
unsigned char repetitions, packet_header, *final;
|
unsigned char repetitions, packet_header, *final;
|
||||||
unsigned short row, col;
|
|
||||||
|
|
||||||
|
|
||||||
/* see line 207 to see why this loop is set up like this */
|
/* see line 207 to see why this loop is set up like this */
|
||||||
for (row = info->start_row; (!info->origin && row) || (info->origin && row < info->height); row += info->increment)
|
for (row = info->start_row; (!info->origin && row) || (info->origin && row < info->height); row += info->increment)
|
||||||
{
|
{
|
||||||
final = (*mpi)->planes[0] + (*mpi)->stride[0] * row;
|
final = mpi->planes[0] + mpi->stride[0] * row;
|
||||||
|
|
||||||
for (col = 0; col < info->width; col += repetitions)
|
for (col = 0; col < info->width; col += repetitions)
|
||||||
{
|
{
|
||||||
packet_header = *data++;
|
packet_header = *data++;
|
||||||
repetitions = 1 + (packet_header & 0x7f);
|
repetitions = (1 + (packet_header & 0x7f));
|
||||||
|
replen = repetitions * num_bytes;
|
||||||
|
|
||||||
if (packet_header & 0x80) /* runlength encoded packet */
|
if (packet_header & 0x80) /* runlength encoded packet */
|
||||||
{
|
{
|
||||||
memcpy(final, data, num_bytes);
|
memcpy(final, data, num_bytes);
|
||||||
|
|
||||||
for (i = num_bytes; i < repetitions; i *= 2)
|
|
||||||
memcpy(&final[i], final, i);
|
|
||||||
|
|
||||||
for (; i < repetitions * num_bytes; i += 3)
|
// Note: this will be slow when DR to vram!
|
||||||
memcpy(&final[i], final, num_bytes);
|
i=num_bytes;
|
||||||
|
while(2*i<=replen){
|
||||||
|
memcpy(final+i,final,i);
|
||||||
|
i*=2;
|
||||||
|
}
|
||||||
|
memcpy(final+i,final,replen-i);
|
||||||
|
data += num_bytes;
|
||||||
}
|
}
|
||||||
else /* raw packet */
|
else /* raw packet */
|
||||||
memcpy(final, data, repetitions * num_bytes);
|
{
|
||||||
|
memcpy(final, data, replen);
|
||||||
data += (packet_header & 0x80) ? num_bytes : repetitions * num_bytes;
|
data += replen;
|
||||||
final += repetitions * num_bytes;
|
}
|
||||||
|
|
||||||
|
final += replen;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static void decode_uncompressed_tga(TGAInfo *info, unsigned char *data, mp_image_t **mpi)
|
static void decode_uncompressed_tga(TGAInfo *info, unsigned char *data, mp_image_t *mpi)
|
||||||
{
|
{
|
||||||
unsigned short row;
|
|
||||||
unsigned char *final;
|
unsigned char *final;
|
||||||
short num_bytes = info->bpp / 8;
|
int row, num_bytes = info->bpp / 8;
|
||||||
|
|
||||||
|
|
||||||
/* see line 207 to see why this loop is set up like this */
|
/* see line 207 to see why this loop is set up like this */
|
||||||
for (row = info->start_row; (!info->origin && row) || (info->origin && row < info->height); row += info->increment)
|
for (row = info->start_row; (!info->origin && row) || (info->origin && row < info->height); row += info->increment)
|
||||||
{
|
{
|
||||||
final = (*mpi)->planes[0] + (*mpi)->stride[0] * row;
|
final = mpi->planes[0] + mpi->stride[0] * row;
|
||||||
memcpy(final, data, info->width * num_bytes);
|
memcpy(final, data, info->width * num_bytes);
|
||||||
data += info->width * num_bytes;
|
data += info->width * num_bytes;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static short read_tga_header(unsigned char *buf, TGAInfo **info)
|
static short read_tga_header(unsigned char *buf, TGAInfo *info)
|
||||||
{
|
{
|
||||||
(*info)->id_len = buf[0];
|
info->id_len = buf[0];
|
||||||
|
|
||||||
(*info)->img_type = buf[2];
|
info->img_type = buf[2];
|
||||||
|
|
||||||
/* targa data is always stored in little endian byte order */
|
/* targa data is always stored in little endian byte order */
|
||||||
(*info)->width = le2me_16(*(unsigned short *) &buf[12]);
|
info->width = le2me_16(*(unsigned short *) &buf[12]);
|
||||||
(*info)->height = le2me_16(*(unsigned short *) &buf[14]);
|
info->height = le2me_16(*(unsigned short *) &buf[14]);
|
||||||
|
|
||||||
(*info)->bpp = buf[16];
|
info->bpp = buf[16];
|
||||||
|
|
||||||
(*info)->origin = (buf[17] & 0x20) >> 5;
|
info->origin = (buf[17] & 0x20) >> 5;
|
||||||
|
|
||||||
/* FIXME check for valid targa data */
|
/* FIXME check for valid targa data */
|
||||||
|
|
||||||
@ -195,7 +183,7 @@ static mp_image_t *decode(sh_video_t *sh, void *raw, int len, int flags)
|
|||||||
if (len <= 0)
|
if (len <= 0)
|
||||||
return NULL; /* skip frame */
|
return NULL; /* skip frame */
|
||||||
|
|
||||||
read_tga_header(data, &info); /* read information about the file */
|
read_tga_header(data, info); /* read information about the file */
|
||||||
|
|
||||||
if (info->bpp == 24)
|
if (info->bpp == 24)
|
||||||
out_fmt = IMGFMT_BGR24;
|
out_fmt = IMGFMT_BGR24;
|
||||||
@ -203,7 +191,7 @@ static mp_image_t *decode(sh_video_t *sh, void *raw, int len, int flags)
|
|||||||
out_fmt = IMGFMT_BGR32;
|
out_fmt = IMGFMT_BGR32;
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
mp_msg(MSGT_DECVIDEO, MSGL_INFO, "Unsupported TGA type!\n");
|
mp_msg(MSGT_DECVIDEO, MSGL_INFO, "Unsupported TGA type! depth=%d\n",info->bpp);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -236,12 +224,11 @@ static mp_image_t *decode(sh_video_t *sh, void *raw, int len, int flags)
|
|||||||
|
|
||||||
/* finally decode the image */
|
/* finally decode the image */
|
||||||
if (info->img_type == TGA_UNCOMP_TRUECOLOR)
|
if (info->img_type == TGA_UNCOMP_TRUECOLOR)
|
||||||
decode_uncompressed_tga(info, data, &mpi);
|
decode_uncompressed_tga(info, data, mpi);
|
||||||
else if (info->img_type == TGA_RLE_TRUECOLOR)
|
else if (info->img_type == TGA_RLE_TRUECOLOR)
|
||||||
decode_rle_tga(info, data, &mpi);
|
decode_rle_tga(info, data, mpi);
|
||||||
else
|
// else
|
||||||
mpi = NULL;
|
// mpi = NULL;
|
||||||
|
|
||||||
|
|
||||||
return mpi;
|
return mpi;
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user