1 /*
2 * Flash Screen Video Version 2 encoder
3 * Copyright (C) 2009 Joshua Warner
4 *
5 * This file is part of FFmpeg.
6 *
7 * FFmpeg is free software; you can redistribute it and/or
8 * modify it under the terms of the GNU Lesser General Public
9 * License as published by the Free Software Foundation; either
10 * version 2.1 of the License, or (at your option) any later version.
11 *
12 * FFmpeg is distributed in the hope that it will be useful,
13 * but WITHOUT ANY WARRANTY; without even the implied warranty of
14 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
15 * Lesser General Public License for more details.
16 *
17 * You should have received a copy of the GNU Lesser General Public
18 * License along with FFmpeg; if not, write to the Free Software
19 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
20 */
21
22 /**
23 * @file
24 * Flash Screen Video Version 2 encoder
25 * @author Joshua Warner
26 */
27
28 /* Differences from version 1 stream:
29 * NOTE: Currently, the only player that supports version 2 streams is Adobe Flash Player itself.
30 * * Supports sending only a range of scanlines in a block,
31 * indicating a difference from the corresponding block in the last keyframe.
32 * * Supports initializing the zlib dictionary with data from the corresponding
33 * block in the last keyframe, to improve compression.
34 * * Supports a hybrid 15-bit rgb / 7-bit palette color space.
35 */
36
37 /* TODO:
38 * Don't keep Block structures for both current frame and keyframe.
39 * Make better heuristics for deciding stream parameters (optimum_* functions). Currently these return constants.
40 * Figure out how to encode palette information in the stream, choose an optimum palette at each keyframe.
41 * Figure out how the zlibPrimeCompressCurrent flag works, implement support.
42 * Find other sample files (that weren't generated here), develop a decoder.
43 */
44
45 #include <stdio.h>
46 #include <stdlib.h>
47 #include <zlib.h>
48
54
55 #define HAS_IFRAME_IMAGE 0x02
56 #define HAS_PALLET_INFO 0x01
57
58 #define COLORSPACE_BGR 0x00
59 #define COLORSPACE_15_7 0x10
60 #define HAS_DIFF_BLOCKS 0x04
61 #define ZLIB_PRIME_COMPRESS_CURRENT 0x02
62 #define ZLIB_PRIME_COMPRESS_PREVIOUS 0x01
63
64 // Disables experimental "smart" parameter-choosing code, as well as the statistics that it depends on.
65 // At the moment, the "smart" code is a great example of how the parameters *shouldn't* be chosen.
67
74
80
85
93
96
101
103
105
107
114 #ifndef FLASHSV2_DUMB
115 double tot_blocks; ///< blocks encoded since last keyframe
116 double diff_blocks; ///< blocks that were different since last keyframe
117 double tot_lines; ///< total scanlines in image since last keyframe
118 double diff_lines; ///< scanlines that were different since last keyframe
119 double raw_size; ///< size of raw frames since last keyframe
120 double comp_size; ///< size of compressed data since last keyframe
121 double uncomp_size; ///< size of uncompressed data since last keyframe
122
123 double total_bits; ///< total bits written to stream so far
124 #endif
126
128 {
135
138 }
139
142 {
143 int row, col;
145 for (col = 0; col < s->
cols; col++) {
146 for (row = 0; row < s->
rows; row++) {
147 b = blocks + (col + row * s->
cols);
151
152 b->
height = (row < s->rows - 1) ?
155
161 databuf += !databuf ? 0 : b->
width * b->
height * 6;
162 }
163 }
164 }
165
167 {
168 #ifndef FLASHSV2_DUMB
169 s->diff_blocks = 0.1;
170 s->tot_blocks = 1;
171 s->diff_lines = 0.1;
172 s->tot_lines = 1;
173 s->raw_size = s->comp_size = s->uncomp_size = 10;
174 #endif
175 }
176
178 {
180
182
188 "Compression level should be 0-9, not %d\n", s->
comp);
189 return -1;
190 }
191
192
193 if ((avctx->
width > 4095) || (avctx->
height > 4095)) {
195 "Input dimensions too large, input must be max 4095x4095 !\n");
196 return -1;
197 }
200 "Input dimensions too small, input must be at least 16x16 !\n");
201 return -1;
202 }
203
205 return -1;
206
207
209
212
215
220
223
226
234
237
241 #ifndef FLASHSV2_DUMB
242 s->total_bits = 1;
243 #endif
244
246 s->
palette_type = -1;
// so that the palette will be generated in reconfigure_at_keyframe
247
253 return -1;
254 }
255
256 return 0;
257 }
258
260 {
261 int i;
264
265 for (i = 0; i < s->
rows * s->
cols; i++) {
270 }
272
273 return 0;
274 }
275
277 {
278 //this isn't implemented yet! Default palette only!
279 return -1;
280 }
281
283 {
286
287 if (buf_size < 5)
288 return -1;
289
291
296
298 buf_pos = 4;
299
300 buf[buf_pos++] = s->
flags;
301
304 if (len < 0)
305 return -1;
307 }
308
309 return buf_pos;
310 }
311
313 {
314 int buf_pos = 0;
316
318 block_size += 2;
320 block_size += 2;
321 if (block_size > 0)
322 block_size += 1;
323 if (buf_size < block_size + 2)
324 return -1;
325
326 buf[buf_pos++] = block_size >> 8;
327 buf[buf_pos++] = block_size;
328
329 if (block_size == 0)
330 return buf_pos;
331
332 buf[buf_pos++] = b->
flags;
333
335 buf[buf_pos++] = (b->
start);
336 buf[buf_pos++] = (b->
len);
337 }
338
340 //This feature of the format is poorly understood, and as of now, unused.
341 buf[buf_pos++] = (b->
col);
342 buf[buf_pos++] = (b->
row);
343 }
344
346
348
349 return buf_pos;
350 }
351
353 {
355 return res == Z_OK ? 0 : -1;
356 }
357
359 int *buf_size,
int comp)
360 {
362 int res;
366 res = deflateInit(&s, comp);
367 if (res < 0)
368 return -1;
369
370 s.next_in = prime->
enc;
372 while (s.avail_in > 0) {
374 s.avail_out = *buf_size;
375 res = deflate(&s, Z_SYNC_FLUSH);
376 if (res < 0)
377 return -1;
378 }
379
383 s.avail_out = *buf_size;
384 res = deflate(&s, Z_FINISH);
385 deflateEnd(&s);
386 *buf_size -= s.avail_out;
387 if (res != Z_STREAM_END)
388 return -1;
389 return 0;
390 }
391
393 {
394 int i;
396 for (i = 0; i < b->
start; i++)
397 memcpy(ptr + i * b->
width * 3, src + i * stride, b->
width * 3);
400 memcpy(ptr + i * b->
width * 3, src + i * stride, b->
width * 3);
402 for (; i < b->
height; i++)
403 memcpy(ptr + i * b->
width * 3, src + i * stride, b->
width * 3);
406 }
407
409 {
410 return (src[0] >> 3) | ((src[1] & 0xf8) << 2) | ((src[2] & 0xf8) << 7);
411 }
412
414 {
415 unsigned int t1 = (c1 & 0x000000ff) + ((c1 & 0x0000ff00) >> 8) + ((c1 & 0x00ff0000) >> 16);
416 unsigned int t2 = (c2 & 0x000000ff) + ((c2 & 0x0000ff00) >> 8) + ((c2 & 0x00ff0000) >> 16);
417
418 return abs(t1 - t2) + abs((c1 & 0x000000ff) - (c2 & 0x000000ff)) +
419 abs(((c1 & 0x0000ff00) >> 8) - ((c2 & 0x0000ff00) >> 8)) +
420 abs(((c1 & 0x00ff0000) >> 16) - ((c2 & 0x00ff0000) >> 16));
421 }
422
424 {
425 return palette->
index[c15];
426 }
427
429 {
430 int i,
min = 0x7fffffff;
431 int minc = -1;
432 for (i = 0; i < 128; i++) {
435 if (diff < min) {
437 minc = i;
438 }
439 }
440 return minc;
441 }
442
444 {
445 return (src[0]) | (src[1] << 8) | (src[2] << 16);
446 }
447
449 int dist)
450 {
456 if (dist + d15 >= d7) {
457 dest[0] = c7;
458 return 1;
459 } else {
460 dest[0] = 0x80 | (c15 >> 8);
461 dest[1] = c15 & 0xff;
462 return 2;
463 }
464 }
465
467 {
469 unsigned int bgr, c15,
index;
470 for (r = 4; r < 256; r += 8) {
471 for (g = 4; g < 256; g += 8) {
472 for (b = 4; b < 256; b += 8) {
473 bgr = b | (g << 8) | (r << 16);
474 c15 = (b >> 3) | ((g & 0xf8) << 2) | ((r & 0xf8) << 7);
476
478 }
479 }
480 }
481 return 0;
482 }
483
485 0x00000000, 0x00333333, 0x00666666, 0x00999999, 0x00CCCCCC, 0x00FFFFFF,
486 0x00330000, 0x00660000, 0x00990000, 0x00CC0000, 0x00FF0000, 0x00003300,
487 0x00006600, 0x00009900, 0x0000CC00, 0x0000FF00, 0x00000033, 0x00000066,
488 0x00000099, 0x000000CC, 0x000000FF, 0x00333300, 0x00666600, 0x00999900,
489 0x00CCCC00, 0x00FFFF00, 0x00003333, 0x00006666, 0x00009999, 0x0000CCCC,
490 0x0000FFFF, 0x00330033, 0x00660066, 0x00990099, 0x00CC00CC, 0x00FF00FF,
491 0x00FFFF33, 0x00FFFF66, 0x00FFFF99, 0x00FFFFCC, 0x00FF33FF, 0x00FF66FF,
492 0x00FF99FF, 0x00FFCCFF, 0x0033FFFF, 0x0066FFFF, 0x0099FFFF, 0x00CCFFFF,
493 0x00CCCC33, 0x00CCCC66, 0x00CCCC99, 0x00CCCCFF, 0x00CC33CC, 0x00CC66CC,
494 0x00CC99CC, 0x00CCFFCC, 0x0033CCCC, 0x0066CCCC, 0x0099CCCC, 0x00FFCCCC,
495 0x00999933, 0x00999966, 0x009999CC, 0x009999FF, 0x00993399, 0x00996699,
496 0x0099CC99, 0x0099FF99, 0x00339999, 0x00669999, 0x00CC9999, 0x00FF9999,
497 0x00666633, 0x00666699, 0x006666CC, 0x006666FF, 0x00663366, 0x00669966,
498 0x0066CC66, 0x0066FF66, 0x00336666, 0x00996666, 0x00CC6666, 0x00FF6666,
499 0x00333366, 0x00333399, 0x003333CC, 0x003333FF, 0x00336633, 0x00339933,
500 0x0033CC33, 0x0033FF33, 0x00663333, 0x00993333, 0x00CC3333, 0x00FF3333,
501 0x00003366, 0x00336600, 0x00660033, 0x00006633, 0x00330066, 0x00663300,
502 0x00336699, 0x00669933, 0x00993366, 0x00339966, 0x00663399, 0x00996633,
503 0x006699CC, 0x0099CC66, 0x00CC6699, 0x0066CC99, 0x009966CC, 0x00CC9966,
504 0x0099CCFF, 0x00CCFF99, 0x00FF99CC, 0x0099FFCC, 0x00CC99FF, 0x00FFCC99,
505 0x00111111, 0x00222222, 0x00444444, 0x00555555, 0x00AAAAAA, 0x00BBBBBB,
506 0x00DDDDDD, 0x00EEEEEE
507 };
508
510 {
513
515 }
516
519 {
520 //this isn't implemented yet! Default palette only!
521 return -1;
522 }
523
526 {
528 for (x = 0; x <
width; x++) {
530 }
532 }
533
536 {
537 int i;
539 for (i = 0; i < b->
start; i++)
545 for (; i < b->
height; i++)
549 }
550
553 int dist, int keyframe)
554 {
557 int res;
558
561 } else {
563 }
564
568 if (res)
569 return res;
570
571 if (!keyframe) {
573 if (res)
574 return res;
575
576 if (buf_size < b->data_size) {
578 memcpy(b->
data, buf, buf_size);
580 }
581 }
582 } else {
584 }
585 return 0;
586 }
587
590 {
591 if (memcmp(src, frame, b->
width * 3) != 0) {
593 memcpy(frame, src, b->
width * 3);
594 #ifndef FLASHSV2_DUMB
595 s->diff_lines++;
596 #endif
597 }
598 if (memcmp(src, key, b->
width * 3) != 0) {
602 }
603 return 0;
604 }
605
607 int keyframe)
608 {
609 int sl, rsl, col, pos, possl;
612 for (col = 0; col < s->
cols; col++) {
619 }
620 }
621 #ifndef FLASHSV2_DUMB
623 #endif
624 return 0;
625 }
626
628 {
629 int row, col, res;
632 for (row = 0; row < s->
rows; row++) {
633 for (col = 0; col < s->
cols; col++) {
637 if (keyframe) {
640 }
else if (!b->
dirty) {
644 continue;
647 }
650 #ifndef FLASHSV2_DUMB
652 s->diff_blocks++;
655 #endif
656 if (res)
657 return res;
658 }
659 }
660 #ifndef FLASHSV2_DUMB
663 #endif
664 return 0;
665 }
666
668 int buf_size)
669 {
670 int row, col, buf_pos = 0,
len;
672 for (row = 0; row < s->
rows; row++) {
673 for (col = 0; col < s->
cols; col++) {
680 }
681 }
682 return buf_pos;
683 }
684
687 {
688 int buf_pos, res;
689
691 if (res)
692 return res;
694 if (res)
695 return res;
696
698 if (res < 0) {
699 return res;
700 } else {
701 buf_pos = res;
702 }
704 if (res < 0)
705 return res;
706 buf_pos += res;
707 #ifndef FLASHSV2_DUMB
708 s->total_bits += ((double) buf_pos) * 8.0;
709 #endif
710
711 return buf_pos;
712 }
713
715 {
716 #ifndef FLASHSV2_DUMB
717 double block_ratio, line_ratio, enc_ratio, comp_ratio, data_ratio;
719 block_ratio = s->diff_blocks / s->tot_blocks;
720 line_ratio = s->diff_lines / s->tot_lines;
721 enc_ratio = s->uncomp_size / s->raw_size;
722 comp_ratio = s->comp_size / s->uncomp_size;
723 data_ratio = s->comp_size / s->raw_size;
724
725 if ((block_ratio >= 0.5 && line_ratio / block_ratio <= 0.5) || line_ratio >= 0.95) {
726 *keyframe = 1;
727 return;
728 }
729 }
730 #else
731 return;
732 #endif
733 }
734
735 #ifndef FLASHSV2_DUMB
736 static const double block_size_fraction = 1.0 / 300;
737 static const double use15_7_threshold = 8192;
738 static const double color15_7_factor = 100;
739 #endif
741 {
742 #ifndef FLASHSV2_DUMB
743 double save = (1-pow(s->diff_lines/s->diff_blocks/s->
block_height, 0.5)) * s->comp_size/s->tot_blocks;
745 int pwidth = ((int)
width);
746 return FFCLIP(pwidth & ~15, 256, 16);
747 #else
748 return 64;
749 #endif
750 }
751
753 {
754 #ifndef FLASHSV2_DUMB
755 double save = (1-pow(s->diff_lines/s->diff_blocks/s->
block_height, 0.5)) * s->comp_size/s->tot_blocks;
757 int pheight = ((int)
height);
758 return FFCLIP(pheight & ~15, 256, 16);
759 #else
760 return 64;
761 #endif
762 }
763
765 {
766 #ifndef FLASHSV2_DUMB
769 if (ideal + use15_7_threshold < s->total_bits) {
770 return 1;
771 } else {
772 return 0;
773 }
774 #else
776 #endif
777 }
778
780 {
781 #ifndef FLASHSV2_DUMB
782 double ideal =
785 int dist = pow((s->total_bits / ideal) * color15_7_factor, 3);
787 return dist;
788 #else
789 return 15;
790 #endif
791 }
792
793
796 {
797 int update_palette = 0;
798 int res;
801
804
813 return -1;
814 }
816 }
819
824 }
825 }
826
831 if (res)
832 return res;
837 if (res)
838 return res;
841 }
842 }
843
844
846
847 return 0;
848 }
849
851 const AVFrame *p,
int *got_packet)
852 {
854 int res;
855 int keyframe = 0;
856
858 return res;
859
860 /* First frame needs to be a keyframe */
862 keyframe = 1;
863
864 /* Check the placement of keyframes */
867 keyframe = 1;
868 }
869
870 if (!keyframe
873 if (keyframe)
875 }
876
877 if (keyframe) {
879 if (res)
880 return res;
881 }
882
885
887
888 if (keyframe) {
893 }
894
896 *got_packet = 1;
897
898 return 0;
899 }
900
902 {
904
906
907 return 0;
908 }
909
920 };