46 #define FLI_256_COLOR 4
54 #define FLI_DTA_BRUN 25
55 #define FLI_DTA_COPY 26
58 #define FLI_TYPE_CODE (0xAF11)
59 #define FLC_FLX_TYPE_CODE (0xAF12)
60 #define FLC_DTA_TYPE_CODE (0xAF44)
61 #define FLC_MAGIC_CARPET_SYNTHETIC_TYPE_CODE (0xAF13)
64 ptrdiff_t
limit,
int direction)
66 if (( direction && ptr + n >
limit) ||
67 (!direction && ptr + n <
limit))
72 #define CHECK_PIXEL_PTR(n) \
74 ret = check_pixel_ptr(pixel_ptr, (n), pixel_limit, direction); \
79 #define CHECK_Y_PTR() \
81 ret = check_pixel_ptr(y_ptr, 0, pixel_limit, direction); \
113 if (
s->avctx->extradata_size == 12) {
121 for (
i = 0;
i < 256;
i++) {
134 s->fli_type =
AV_RL16(&fli_header[4]);
135 depth =
AV_RL16(&fli_header[12]);
153 av_log(avctx,
AV_LOG_ERROR,
"Unknown FLC/FLX depth of %d Bpp is unsupported.\n",depth);
167 AVFrame *rframe,
int *got_frame,
168 const uint8_t *buf,
int buf_size)
178 unsigned int chunk_size;
181 int i, j,
ret, direction;
184 int compressed_lines;
191 unsigned char *pixels;
192 ptrdiff_t pixel_limit;
199 direction =
s->frame->linesize[0] > 0;
200 pixels =
s->frame->data[0];
201 pixel_limit =
s->avctx->height *
s->frame->linesize[0];
208 num_chunks = bytestream2_get_le16(&g2);
217 while ((
frame_size >= 6) && (num_chunks > 0) &&
219 int stream_ptr_after_chunk;
220 chunk_size = bytestream2_get_le32(&g2);
223 "Invalid chunk_size = %u > frame_size = %u\n", chunk_size,
frame_size);
228 chunk_type = bytestream2_get_le16(&g2);
230 switch (chunk_type) {
235 for (lines = 0; lines <
s->avctx->height; lines++) {
240 pixel_countdown = (
s->avctx->width + 7) >> 3;
241 while (pixel_countdown > 0) {
244 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
251 int value = bytestream2_get_byte(&g2);
253 for (j = 0; j < byte_run; j++) {
254 pixels[pixel_ptr++] =
value;
256 if (pixel_countdown < 0)
258 pixel_countdown, lines);
261 byte_run = -byte_run;
265 for (j = 0; j < byte_run; j++) {
266 pixels[pixel_ptr++] = bytestream2_get_byte(&g2);
268 if (pixel_countdown < 0)
270 pixel_countdown, lines);
275 y_ptr +=
s->frame->linesize[0];
281 starting_line = bytestream2_get_le16(&g2);
282 if (starting_line >=
s->avctx->height)
285 y_ptr += starting_line *
s->frame->linesize[0];
287 compressed_lines = bytestream2_get_le16(&g2);
288 while (compressed_lines > 0) {
291 pixel_countdown = (
s->avctx->width + 7) >> 3;
294 line_packets = bytestream2_get_byte(&g2);
295 if (line_packets > 0) {
296 for (
i = 0;
i < line_packets;
i++) {
300 pixel_skip = bytestream2_get_byte(&g2);
301 pixel_ptr += pixel_skip;
302 pixel_countdown -= pixel_skip;
303 byte_run =
sign_extend(bytestream2_get_byte(&g2),8);
308 for (j = 0; j < byte_run; j++, pixel_countdown--) {
309 pixels[pixel_ptr++] = bytestream2_get_byte(&g2);
311 }
else if (byte_run < 0) {
312 int value = bytestream2_get_byte(&g2);
313 byte_run = -byte_run;
315 for (j = 0; j < byte_run; j++, pixel_countdown--) {
316 pixels[pixel_ptr++] =
value;
322 y_ptr +=
s->frame->linesize[0];
347 "and final chunk ptr = %d\n", buf_size,
359 AVFrame *rframe,
int *got_frame,
360 const uint8_t *buf,
int buf_size)
367 unsigned char palette_idx1;
368 unsigned char palette_idx2;
373 unsigned int chunk_size;
376 int i, j,
ret, direction;
381 unsigned char r,
g,
b;
384 int compressed_lines;
391 unsigned char *pixels;
392 ptrdiff_t pixel_limit;
399 direction =
s->frame->linesize[0] > 0;
400 pixels =
s->frame->data[0];
401 pixel_limit =
s->avctx->height *
s->frame->linesize[0];
408 num_chunks = bytestream2_get_le16(&g2);
417 while ((
frame_size >= 6) && (num_chunks > 0) &&
419 int stream_ptr_after_chunk;
420 chunk_size = bytestream2_get_le32(&g2);
423 "Invalid chunk_size = %u > frame_size = %u\n", chunk_size,
frame_size);
428 chunk_type = bytestream2_get_le16(&g2);
430 switch (chunk_type) {
442 color_packets = bytestream2_get_le16(&g2);
444 for (
i = 0;
i < color_packets;
i++) {
446 palette_ptr += bytestream2_get_byte(&g2);
449 color_changes = bytestream2_get_byte(&g2);
452 if (color_changes == 0)
458 for (j = 0; j < color_changes; j++) {
462 if ((
unsigned)palette_ptr >= 256)
465 r = bytestream2_get_byte(&g2) << color_shift;
466 g = bytestream2_get_byte(&g2) << color_shift;
467 b = bytestream2_get_byte(&g2) << color_shift;
468 entry = 0xFF
U << 24 |
r << 16 |
g << 8 |
b;
469 if (color_shift == 2)
471 if (
s->palette[palette_ptr] !=
entry)
473 s->palette[palette_ptr++] =
entry;
480 compressed_lines = bytestream2_get_le16(&g2);
481 while (compressed_lines > 0) {
485 line_packets =
sign_extend(bytestream2_get_le16(&g2), 16);
486 if ((line_packets & 0xC000) == 0xC000) {
488 line_packets = -line_packets;
489 if (line_packets >
s->avctx->height)
491 y_ptr += line_packets *
s->frame->linesize[0];
492 }
else if ((line_packets & 0xC000) == 0x4000) {
494 }
else if ((line_packets & 0xC000) == 0x8000) {
496 pixel_ptr= y_ptr +
s->frame->linesize[0] - 1;
498 pixels[pixel_ptr] = line_packets & 0xff;
503 pixel_countdown =
s->avctx->width;
504 for (
i = 0;
i < line_packets;
i++) {
508 pixel_skip = bytestream2_get_byte(&g2);
509 pixel_ptr += pixel_skip;
510 pixel_countdown -= pixel_skip;
511 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
513 byte_run = -byte_run;
514 palette_idx1 = bytestream2_get_byte(&g2);
515 palette_idx2 = bytestream2_get_byte(&g2);
517 for (j = 0; j < byte_run; j++, pixel_countdown -= 2) {
518 pixels[pixel_ptr++] = palette_idx1;
519 pixels[pixel_ptr++] = palette_idx2;
525 for (j = 0; j < byte_run * 2; j++, pixel_countdown--) {
526 pixels[pixel_ptr++] = bytestream2_get_byte(&g2);
531 y_ptr +=
s->frame->linesize[0];
538 starting_line = bytestream2_get_le16(&g2);
539 if (starting_line >=
s->avctx->height)
542 y_ptr += starting_line *
s->frame->linesize[0];
544 compressed_lines = bytestream2_get_le16(&g2);
545 while (compressed_lines > 0) {
548 pixel_countdown =
s->avctx->width;
551 line_packets = bytestream2_get_byte(&g2);
552 if (line_packets > 0) {
553 for (
i = 0;
i < line_packets;
i++) {
557 pixel_skip = bytestream2_get_byte(&g2);
558 pixel_ptr += pixel_skip;
559 pixel_countdown -= pixel_skip;
560 byte_run =
sign_extend(bytestream2_get_byte(&g2),8);
565 for (j = 0; j < byte_run; j++, pixel_countdown--) {
566 pixels[pixel_ptr++] = bytestream2_get_byte(&g2);
568 }
else if (byte_run < 0) {
569 byte_run = -byte_run;
570 palette_idx1 = bytestream2_get_byte(&g2);
572 for (j = 0; j < byte_run; j++, pixel_countdown--) {
573 pixels[pixel_ptr++] = palette_idx1;
579 y_ptr +=
s->frame->linesize[0];
586 for (
int y = 0; y <
s->avctx->height; y++)
587 memset(pixels + y *
s->frame->linesize[0], 0,
s->avctx->width);
594 for (lines = 0; lines <
s->avctx->height; lines++) {
599 pixel_countdown =
s->avctx->width;
600 while (pixel_countdown > 0) {
603 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
610 palette_idx1 = bytestream2_get_byte(&g2);
612 for (j = 0; j < byte_run; j++) {
613 pixels[pixel_ptr++] = palette_idx1;
615 if (pixel_countdown < 0)
617 pixel_countdown, lines);
620 byte_run = -byte_run;
624 for (j = 0; j < byte_run; j++) {
625 pixels[pixel_ptr++] = bytestream2_get_byte(&g2);
627 if (pixel_countdown < 0)
629 pixel_countdown, lines);
634 y_ptr +=
s->frame->linesize[0];
640 if (chunk_size - 6 !=
FFALIGN(
s->avctx->width, 4) *
s->avctx->height) {
642 "has incorrect size, skipping chunk\n", chunk_size - 6);
645 for (y_ptr = 0;
check_pixel_ptr(y_ptr,
s->avctx->width, pixel_limit, direction) == 0;
646 y_ptr +=
s->frame->linesize[0]) {
649 if (
s->avctx->width & 3)
679 "and final chunk ptr = %d\n", buf_size,
684 if (
s->new_palette) {
697 AVFrame *rframe,
int *got_frame,
698 const uint8_t *buf,
int buf_size)
706 unsigned char palette_idx1;
711 unsigned int chunk_size;
714 int i, j,
ret, direction;
717 int compressed_lines;
723 unsigned char *pixels;
725 ptrdiff_t pixel_limit;
732 direction =
s->frame->linesize[0] > 0;
733 pixels =
s->frame->data[0];
734 pixel_limit =
s->avctx->height *
s->frame->linesize[0];
738 num_chunks = bytestream2_get_le16(&g2);
748 while ((
frame_size > 0) && (num_chunks > 0) &&
750 int stream_ptr_after_chunk;
751 chunk_size = bytestream2_get_le32(&g2);
754 "Invalid chunk_size = %u > frame_size = %u\n", chunk_size,
frame_size);
759 chunk_type = bytestream2_get_le16(&g2);
762 switch (chunk_type) {
769 "Unexpected Palette chunk %d in non-palettized FLC\n",
777 compressed_lines = bytestream2_get_le16(&g2);
778 while (compressed_lines > 0) {
782 line_packets =
sign_extend(bytestream2_get_le16(&g2), 16);
783 if (line_packets < 0) {
784 line_packets = -line_packets;
785 if (line_packets >
s->avctx->height)
787 y_ptr += line_packets *
s->frame->linesize[0];
792 pixel_countdown =
s->avctx->width;
793 for (
i = 0;
i < line_packets;
i++) {
797 pixel_skip = bytestream2_get_byte(&g2);
798 pixel_ptr += (pixel_skip*2);
799 pixel_countdown -= pixel_skip;
800 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
802 byte_run = -byte_run;
803 pixel = bytestream2_get_le16(&g2);
805 for (j = 0; j < byte_run; j++, pixel_countdown -= 2) {
806 *((
signed short*)(&pixels[pixel_ptr])) =
pixel;
813 for (j = 0; j < byte_run; j++, pixel_countdown--) {
814 *((
signed short*)(&pixels[pixel_ptr])) = bytestream2_get_le16(&g2);
820 y_ptr +=
s->frame->linesize[0];
832 for (
int y = 0; y <
s->avctx->height; y++)
833 memset(pixels + y *
s->frame->linesize[0], 0,
s->avctx->width * 2);
838 for (lines = 0; lines <
s->avctx->height; lines++) {
843 pixel_countdown = (
s->avctx->width * 2);
845 while (pixel_countdown > 0) {
848 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
850 palette_idx1 = bytestream2_get_byte(&g2);
852 for (j = 0; j < byte_run; j++) {
853 pixels[pixel_ptr++] = palette_idx1;
855 if (pixel_countdown < 0)
857 pixel_countdown, lines);
860 byte_run = -byte_run;
864 for (j = 0; j < byte_run; j++) {
865 palette_idx1 = bytestream2_get_byte(&g2);
866 pixels[pixel_ptr++] = palette_idx1;
868 if (pixel_countdown < 0)
870 pixel_countdown, lines);
882 pixel_countdown =
s->avctx->width;
883 while (pixel_countdown > 0) {
884 *((
signed short*)(&pixels[pixel_ptr])) =
AV_RL16(&buf[pixel_ptr]);
888 y_ptr +=
s->frame->linesize[0];
894 for (lines = 0; lines <
s->avctx->height; lines++) {
899 pixel_countdown =
s->avctx->width;
901 while (pixel_countdown > 0) {
904 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
906 pixel = bytestream2_get_le16(&g2);
908 for (j = 0; j < byte_run; j++) {
909 *((
signed short*)(&pixels[pixel_ptr])) =
pixel;
912 if (pixel_countdown < 0)
917 byte_run = -byte_run;
921 for (j = 0; j < byte_run; j++) {
922 *((
signed short*)(&pixels[pixel_ptr])) = bytestream2_get_le16(&g2);
925 if (pixel_countdown < 0)
932 y_ptr +=
s->frame->linesize[0];
939 if (chunk_size - 6 > (
unsigned int)(
FFALIGN(
s->avctx->width, 2) *
s->avctx->height)*2) {
941 "bigger than image, skipping chunk\n", chunk_size - 6);
947 for (y_ptr = 0;
check_pixel_ptr(y_ptr, 2*
s->avctx->width, pixel_limit, direction) == 0;
948 y_ptr +=
s->frame->linesize[0]) {
950 pixel_countdown =
s->avctx->width;
952 while (pixel_countdown > 0) {
953 *((
signed short*)(&pixels[y_ptr + pixel_ptr])) = bytestream2_get_le16(&g2);
957 if (
s->avctx->width & 1)
999 AVFrame *rframe,
int *got_frame,
1000 const uint8_t *buf,
int buf_size)
1005 ptrdiff_t pixel_ptr;
1006 unsigned char palette_idx1;
1011 unsigned int chunk_size;
1014 int i, j,
ret, direction;
1017 int compressed_lines;
1022 int pixel_countdown;
1023 unsigned char *pixels;
1025 ptrdiff_t pixel_limit;
1032 direction =
s->frame->linesize[0] > 0;
1033 pixels =
s->frame->data[0];
1034 pixel_limit =
s->avctx->height *
s->frame->linesize[0];
1038 num_chunks = bytestream2_get_le16(&g2);
1048 while ((
frame_size > 0) && (num_chunks > 0) &&
1050 int stream_ptr_after_chunk;
1051 chunk_size = bytestream2_get_le32(&g2);
1054 "Invalid chunk_size = %u > frame_size = %u\n", chunk_size,
frame_size);
1059 chunk_type = bytestream2_get_le16(&g2);
1062 switch (chunk_type) {
1069 "Unexpected Palette chunk %d in non-palettized FLC\n",
1077 compressed_lines = bytestream2_get_le16(&g2);
1078 while (compressed_lines > 0) {
1082 line_packets =
sign_extend(bytestream2_get_le16(&g2), 16);
1083 if (line_packets < 0) {
1084 line_packets = -line_packets;
1085 if (line_packets >
s->avctx->height)
1087 y_ptr += line_packets *
s->frame->linesize[0];
1092 pixel_countdown =
s->avctx->width;
1093 for (
i = 0;
i < line_packets;
i++) {
1097 pixel_skip = bytestream2_get_byte(&g2);
1098 pixel_ptr += (pixel_skip*3);
1099 pixel_countdown -= pixel_skip;
1100 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
1102 byte_run = -byte_run;
1103 pixel = bytestream2_get_le24(&g2);
1105 for (j = 0; j < byte_run; j++, pixel_countdown -= 1) {
1113 for (j = 0; j < byte_run; j++, pixel_countdown--) {
1114 pixel = bytestream2_get_le24(&g2);
1121 y_ptr +=
s->frame->linesize[0];
1133 for (
int y = 0; y <
s->avctx->height; y++)
1134 memset(pixels + y *
s->frame->linesize[0], 0,
s->avctx->width * 3);
1139 for (lines = 0; lines <
s->avctx->height; lines++) {
1144 pixel_countdown = (
s->avctx->width * 3);
1146 while (pixel_countdown > 0) {
1149 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
1151 palette_idx1 = bytestream2_get_byte(&g2);
1153 for (j = 0; j < byte_run; j++) {
1154 pixels[pixel_ptr++] = palette_idx1;
1156 if (pixel_countdown < 0)
1158 pixel_countdown, lines);
1161 byte_run = -byte_run;
1165 for (j = 0; j < byte_run; j++) {
1166 palette_idx1 = bytestream2_get_byte(&g2);
1167 pixels[pixel_ptr++] = palette_idx1;
1169 if (pixel_countdown < 0)
1171 pixel_countdown, lines);
1176 y_ptr +=
s->frame->linesize[0];
1182 for (lines = 0; lines <
s->avctx->height; lines++) {
1187 pixel_countdown =
s->avctx->width;
1189 while (pixel_countdown > 0) {
1192 byte_run =
sign_extend(bytestream2_get_byte(&g2), 8);
1194 pixel = bytestream2_get_le24(&g2);
1196 for (j = 0; j < byte_run; j++) {
1200 if (pixel_countdown < 0)
1205 byte_run = -byte_run;
1209 for (j = 0; j < byte_run; j++) {
1210 pixel = bytestream2_get_le24(&g2);
1214 if (pixel_countdown < 0)
1221 y_ptr +=
s->frame->linesize[0];
1228 if (chunk_size - 6 > (
unsigned int)(
FFALIGN(
s->avctx->width, 2) *
s->avctx->height)*3) {
1230 "bigger than image, skipping chunk\n", chunk_size - 6);
1233 for (y_ptr = 0;
check_pixel_ptr(y_ptr, 3*
s->avctx->width, pixel_limit, direction) == 0;
1234 y_ptr +=
s->frame->linesize[0]) {
1237 if (
s->avctx->width & 1)
1281 const uint8_t *buf = avpkt->
data;
1282 int buf_size = avpkt->
size;
1302 av_log(avctx,
AV_LOG_ERROR,
"Unknown FLC format, my science cannot explain how this happened.\n");