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
49 #include "libavutil/imgutils.h"
50 #include "avcodec.h"
51 #include "internal.h"
52 #include "put_bits.h"
53 #include "bytestream.h"
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.
66 #define FLASHSV2_DUMB
67
68 typedef struct Block {
69 uint8_t *enc;
70 uint8_t *sl_begin, *sl_end;
71 int enc_size;
72 uint8_t *data;
73 unsigned long data_size;
74
75 uint8_t start, len;
76 uint8_t dirty;
77 uint8_t col, row, width, height;
78 uint8_t flags;
79 } Block;
80
81 typedef struct Palette {
82 unsigned colors[128];
83 uint8_t index[1 << 15];
84 } Palette;
85
86 typedef struct FlashSV2Context {
87 AVCodecContext *avctx;
88 uint8_t *current_frame;
89 uint8_t *key_frame;
90 uint8_t *encbuffer;
91 uint8_t *keybuffer;
92 uint8_t *databuffer;
93
94 uint8_t *blockbuffer;
95 int blockbuffer_size;
96
97 Block *frame_blocks;
98 Block *key_blocks;
99 int frame_size;
100 int blocks_size;
101
102 int use15_7, dist, comp;
103
104 int rows, cols;
105
106 int last_key_frame;
107
108 int image_width, image_height;
109 int block_width, block_height;
110 uint8_t flags;
111 uint8_t use_custom_palette;
112 uint8_t palette_type; ///< 0=>default, 1=>custom - changed when palette regenerated.
113 Palette palette;
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
125 } FlashSV2Context;
126
cleanup(FlashSV2Context * s)127 static av_cold void cleanup(FlashSV2Context * s)
128 {
129 av_freep(&s->encbuffer);
130 av_freep(&s->keybuffer);
131 av_freep(&s->databuffer);
132 av_freep(&s->blockbuffer);
133 av_freep(&s->current_frame);
134 av_freep(&s->key_frame);
135
136 av_freep(&s->frame_blocks);
137 av_freep(&s->key_blocks);
138 }
139
init_blocks(FlashSV2Context * s,Block * blocks,uint8_t * encbuf,uint8_t * databuf)140 static void init_blocks(FlashSV2Context * s, Block * blocks,
141 uint8_t * encbuf, uint8_t * databuf)
142 {
143 int row, col;
144 Block *b;
145 memset(blocks, 0, s->cols * s->rows * sizeof(*blocks));
146 for (col = 0; col < s->cols; col++) {
147 for (row = 0; row < s->rows; row++) {
148 b = blocks + (col + row * s->cols);
149 b->width = (col < s->cols - 1) ?
150 s->block_width :
151 s->image_width - col * s->block_width;
152
153 b->height = (row < s->rows - 1) ?
154 s->block_height :
155 s->image_height - row * s->block_height;
156
157 b->row = row;
158 b->col = col;
159 b->enc = encbuf;
160 b->data = databuf;
161 encbuf += b->width * b->height * 3;
162 databuf = databuf ? databuf + b->width * b->height * 6 : NULL;
163 }
164 }
165 }
166
reset_stats(FlashSV2Context * s)167 static void reset_stats(FlashSV2Context * s)
168 {
169 #ifndef FLASHSV2_DUMB
170 s->diff_blocks = 0.1;
171 s->tot_blocks = 1;
172 s->diff_lines = 0.1;
173 s->tot_lines = 1;
174 s->raw_size = s->comp_size = s->uncomp_size = 10;
175 #endif
176 }
177
update_block_dimensions(FlashSV2Context * s,int block_width,int block_height)178 static int update_block_dimensions(FlashSV2Context *s, int block_width, int block_height)
179 {
180 s->block_width = block_width;
181 s->block_height = block_height;
182 s->rows = (s->image_height + s->block_height - 1) / s->block_height;
183 s->cols = (s->image_width + s->block_width - 1) / s->block_width;
184 if (s->rows * s->cols > s->blocks_size / sizeof(Block)) {
185 s->frame_blocks = av_realloc_array(s->frame_blocks, s->rows, s->cols * sizeof(Block));
186 s->key_blocks = av_realloc_array(s->key_blocks, s->cols, s->rows * sizeof(Block));
187 if (!s->frame_blocks || !s->key_blocks) {
188 av_log(s->avctx, AV_LOG_ERROR, "Memory allocation failed.\n");
189 return AVERROR(ENOMEM);
190 }
191 s->blocks_size = s->rows * s->cols * sizeof(Block);
192 }
193 init_blocks(s, s->frame_blocks, s->encbuffer, s->databuffer);
194 init_blocks(s, s->key_blocks, s->keybuffer, 0);
195
196 av_fast_malloc(&s->blockbuffer, &s->blockbuffer_size, block_width * block_height * 6);
197 if (!s->blockbuffer) {
198 av_log(s->avctx, AV_LOG_ERROR, "Could not allocate block buffer.\n");
199 return AVERROR(ENOMEM);
200 }
201 return 0;
202 }
203
204
flashsv2_encode_init(AVCodecContext * avctx)205 static av_cold int flashsv2_encode_init(AVCodecContext * avctx)
206 {
207 FlashSV2Context *s = avctx->priv_data;
208 int ret;
209
210 s->avctx = avctx;
211
212 s->comp = avctx->compression_level;
213 if (s->comp == -1)
214 s->comp = 9;
215 if (s->comp < 0 || s->comp > 9) {
216 av_log(avctx, AV_LOG_ERROR,
217 "Compression level should be 0-9, not %d\n", s->comp);
218 return AVERROR(EINVAL);
219 }
220
221
222 if ((avctx->width > 4095) || (avctx->height > 4095)) {
223 av_log(avctx, AV_LOG_ERROR,
224 "Input dimensions too large, input must be max 4095x4095 !\n");
225 return AVERROR(EINVAL);
226 }
227 if ((avctx->width < 16) || (avctx->height < 16)) {
228 av_log(avctx, AV_LOG_ERROR,
229 "Input dimensions too small, input must be at least 16x16 !\n");
230 return AVERROR(EINVAL);
231 }
232
233 if ((ret = av_image_check_size(avctx->width, avctx->height, 0, avctx)) < 0)
234 return ret;
235
236
237 s->last_key_frame = 0;
238
239 s->image_width = avctx->width;
240 s->image_height = avctx->height;
241
242 s->frame_size = s->image_width * s->image_height * 3;
243
244 s->encbuffer = av_mallocz(s->frame_size);
245 s->keybuffer = av_mallocz(s->frame_size);
246 s->databuffer = av_mallocz(s->frame_size * 6);
247 s->current_frame = av_mallocz(s->frame_size);
248 s->key_frame = av_mallocz(s->frame_size);
249 if (!s->encbuffer || !s->keybuffer || !s->databuffer
250 || !s->current_frame || !s->key_frame) {
251 av_log(avctx, AV_LOG_ERROR, "Memory allocation failed.\n");
252 return AVERROR(ENOMEM);
253 }
254
255 reset_stats(s);
256 #ifndef FLASHSV2_DUMB
257 s->total_bits = 1;
258 #endif
259
260 s->use_custom_palette = 0;
261 s->palette_type = -1; // so that the palette will be generated in reconfigure_at_keyframe
262
263 return update_block_dimensions(s, 64, 64);
264 }
265
new_key_frame(FlashSV2Context * s)266 static int new_key_frame(FlashSV2Context * s)
267 {
268 int i;
269 memcpy(s->key_blocks, s->frame_blocks, s->blocks_size);
270 memcpy(s->key_frame, s->current_frame, s->frame_size);
271
272 for (i = 0; i < s->rows * s->cols; i++) {
273 s->key_blocks[i].enc += (s->keybuffer - s->encbuffer);
274 s->key_blocks[i].sl_begin = 0;
275 s->key_blocks[i].sl_end = 0;
276 s->key_blocks[i].data = 0;
277 }
278 memcpy(s->keybuffer, s->encbuffer, s->frame_size);
279
280 return 0;
281 }
282
write_palette(FlashSV2Context * s,uint8_t * buf,int buf_size)283 static int write_palette(FlashSV2Context * s, uint8_t * buf, int buf_size)
284 {
285 //this isn't implemented yet! Default palette only!
286 return -1;
287 }
288
write_header(FlashSV2Context * s,uint8_t * buf,int buf_size)289 static int write_header(FlashSV2Context * s, uint8_t * buf, int buf_size)
290 {
291 PutBitContext pb;
292 int buf_pos, len;
293
294 if (buf_size < 5)
295 return -1;
296
297 init_put_bits(&pb, buf, buf_size);
298
299 put_bits(&pb, 4, (s->block_width >> 4) - 1);
300 put_bits(&pb, 12, s->image_width);
301 put_bits(&pb, 4, (s->block_height >> 4) - 1);
302 put_bits(&pb, 12, s->image_height);
303
304 flush_put_bits(&pb);
305 buf_pos = 4;
306
307 buf[buf_pos++] = s->flags;
308
309 if (s->flags & HAS_PALLET_INFO) {
310 len = write_palette(s, buf + buf_pos, buf_size - buf_pos);
311 if (len < 0)
312 return -1;
313 buf_pos += len;
314 }
315
316 return buf_pos;
317 }
318
write_block(Block * b,uint8_t * buf,int buf_size)319 static int write_block(Block * b, uint8_t * buf, int buf_size)
320 {
321 int buf_pos = 0;
322 unsigned block_size = b->data_size;
323
324 if (b->flags & HAS_DIFF_BLOCKS)
325 block_size += 2;
326 if (b->flags & ZLIB_PRIME_COMPRESS_CURRENT)
327 block_size += 2;
328 if (block_size > 0)
329 block_size += 1;
330 if (buf_size < block_size + 2)
331 return -1;
332
333 buf[buf_pos++] = block_size >> 8;
334 buf[buf_pos++] = block_size;
335
336 if (block_size == 0)
337 return buf_pos;
338
339 buf[buf_pos++] = b->flags;
340
341 if (b->flags & HAS_DIFF_BLOCKS) {
342 buf[buf_pos++] = (b->start);
343 buf[buf_pos++] = (b->len);
344 }
345
346 if (b->flags & ZLIB_PRIME_COMPRESS_CURRENT) {
347 //This feature of the format is poorly understood, and as of now, unused.
348 buf[buf_pos++] = (b->col);
349 buf[buf_pos++] = (b->row);
350 }
351
352 memcpy(buf + buf_pos, b->data, b->data_size);
353
354 buf_pos += b->data_size;
355
356 return buf_pos;
357 }
358
encode_zlib(Block * b,uint8_t * buf,unsigned long * buf_size,int comp)359 static int encode_zlib(Block * b, uint8_t * buf, unsigned long *buf_size, int comp)
360 {
361 int res = compress2(buf, buf_size, b->sl_begin, b->sl_end - b->sl_begin, comp);
362 return res == Z_OK ? 0 : -1;
363 }
364
encode_zlibprime(Block * b,Block * prime,uint8_t * buf,int * buf_size,int comp)365 static int encode_zlibprime(Block * b, Block * prime, uint8_t * buf,
366 int *buf_size, int comp)
367 {
368 z_stream s;
369 int res;
370 s.zalloc = NULL;
371 s.zfree = NULL;
372 s.opaque = NULL;
373 res = deflateInit(&s, comp);
374 if (res < 0)
375 return -1;
376
377 s.next_in = prime->enc;
378 s.avail_in = prime->enc_size;
379 while (s.avail_in > 0) {
380 s.next_out = buf;
381 s.avail_out = *buf_size;
382 res = deflate(&s, Z_SYNC_FLUSH);
383 if (res < 0)
384 return -1;
385 }
386
387 s.next_in = b->sl_begin;
388 s.avail_in = b->sl_end - b->sl_begin;
389 s.next_out = buf;
390 s.avail_out = *buf_size;
391 res = deflate(&s, Z_FINISH);
392 deflateEnd(&s);
393 *buf_size -= s.avail_out;
394 if (res != Z_STREAM_END)
395 return -1;
396 return 0;
397 }
398
encode_bgr(Block * b,const uint8_t * src,int stride)399 static int encode_bgr(Block * b, const uint8_t * src, int stride)
400 {
401 int i;
402 uint8_t *ptr = b->enc;
403 for (i = 0; i < b->start; i++)
404 memcpy(ptr + i * b->width * 3, src + i * stride, b->width * 3);
405 b->sl_begin = ptr + i * b->width * 3;
406 for (; i < b->start + b->len; i++)
407 memcpy(ptr + i * b->width * 3, src + i * stride, b->width * 3);
408 b->sl_end = ptr + i * b->width * 3;
409 for (; i < b->height; i++)
410 memcpy(ptr + i * b->width * 3, src + i * stride, b->width * 3);
411 b->enc_size = ptr + i * b->width * 3 - b->enc;
412 return b->enc_size;
413 }
414
pixel_color15(const uint8_t * src)415 static inline unsigned pixel_color15(const uint8_t * src)
416 {
417 return (src[0] >> 3) | ((src[1] & 0xf8) << 2) | ((src[2] & 0xf8) << 7);
418 }
419
chroma_diff(unsigned int c1,unsigned int c2)420 static inline unsigned int chroma_diff(unsigned int c1, unsigned int c2)
421 {
422 #define ABSDIFF(a,b) (abs((int)(a)-(int)(b)))
423
424 unsigned int t1 = (c1 & 0x000000ff) + ((c1 & 0x0000ff00) >> 8) + ((c1 & 0x00ff0000) >> 16);
425 unsigned int t2 = (c2 & 0x000000ff) + ((c2 & 0x0000ff00) >> 8) + ((c2 & 0x00ff0000) >> 16);
426
427 return ABSDIFF(t1, t2) + ABSDIFF(c1 & 0x000000ff, c2 & 0x000000ff) +
428 ABSDIFF((c1 & 0x0000ff00) >> 8 , (c2 & 0x0000ff00) >> 8) +
429 ABSDIFF((c1 & 0x00ff0000) >> 16, (c2 & 0x00ff0000) >> 16);
430 }
431
pixel_color7_fast(Palette * palette,unsigned c15)432 static inline int pixel_color7_fast(Palette * palette, unsigned c15)
433 {
434 return palette->index[c15];
435 }
436
pixel_color7_slow(Palette * palette,unsigned color)437 static int pixel_color7_slow(Palette * palette, unsigned color)
438 {
439 int i, min = 0x7fffffff;
440 int minc = -1;
441 for (i = 0; i < 128; i++) {
442 int c1 = palette->colors[i];
443 int diff = chroma_diff(c1, color);
444 if (diff < min) {
445 min = diff;
446 minc = i;
447 }
448 }
449 return minc;
450 }
451
pixel_bgr(const uint8_t * src)452 static inline unsigned pixel_bgr(const uint8_t * src)
453 {
454 return (src[0]) | (src[1] << 8) | (src[2] << 16);
455 }
456
write_pixel_15_7(Palette * palette,uint8_t * dest,const uint8_t * src,int dist)457 static int write_pixel_15_7(Palette * palette, uint8_t * dest, const uint8_t * src,
458 int dist)
459 {
460 unsigned c15 = pixel_color15(src);
461 unsigned color = pixel_bgr(src);
462 int d15 = chroma_diff(color, color & 0x00f8f8f8);
463 int c7 = pixel_color7_fast(palette, c15);
464 int d7 = chroma_diff(color, palette->colors[c7]);
465 if (dist + d15 >= d7) {
466 dest[0] = c7;
467 return 1;
468 } else {
469 dest[0] = 0x80 | (c15 >> 8);
470 dest[1] = c15 & 0xff;
471 return 2;
472 }
473 }
474
update_palette_index(Palette * palette)475 static int update_palette_index(Palette * palette)
476 {
477 int r, g, b;
478 unsigned int bgr, c15, index;
479 for (r = 4; r < 256; r += 8) {
480 for (g = 4; g < 256; g += 8) {
481 for (b = 4; b < 256; b += 8) {
482 bgr = b | (g << 8) | (r << 16);
483 c15 = (b >> 3) | ((g & 0xf8) << 2) | ((r & 0xf8) << 7);
484 index = pixel_color7_slow(palette, bgr);
485
486 palette->index[c15] = index;
487 }
488 }
489 }
490 return 0;
491 }
492
493 static const unsigned int default_screen_video_v2_palette[128] = {
494 0x00000000, 0x00333333, 0x00666666, 0x00999999, 0x00CCCCCC, 0x00FFFFFF,
495 0x00330000, 0x00660000, 0x00990000, 0x00CC0000, 0x00FF0000, 0x00003300,
496 0x00006600, 0x00009900, 0x0000CC00, 0x0000FF00, 0x00000033, 0x00000066,
497 0x00000099, 0x000000CC, 0x000000FF, 0x00333300, 0x00666600, 0x00999900,
498 0x00CCCC00, 0x00FFFF00, 0x00003333, 0x00006666, 0x00009999, 0x0000CCCC,
499 0x0000FFFF, 0x00330033, 0x00660066, 0x00990099, 0x00CC00CC, 0x00FF00FF,
500 0x00FFFF33, 0x00FFFF66, 0x00FFFF99, 0x00FFFFCC, 0x00FF33FF, 0x00FF66FF,
501 0x00FF99FF, 0x00FFCCFF, 0x0033FFFF, 0x0066FFFF, 0x0099FFFF, 0x00CCFFFF,
502 0x00CCCC33, 0x00CCCC66, 0x00CCCC99, 0x00CCCCFF, 0x00CC33CC, 0x00CC66CC,
503 0x00CC99CC, 0x00CCFFCC, 0x0033CCCC, 0x0066CCCC, 0x0099CCCC, 0x00FFCCCC,
504 0x00999933, 0x00999966, 0x009999CC, 0x009999FF, 0x00993399, 0x00996699,
505 0x0099CC99, 0x0099FF99, 0x00339999, 0x00669999, 0x00CC9999, 0x00FF9999,
506 0x00666633, 0x00666699, 0x006666CC, 0x006666FF, 0x00663366, 0x00669966,
507 0x0066CC66, 0x0066FF66, 0x00336666, 0x00996666, 0x00CC6666, 0x00FF6666,
508 0x00333366, 0x00333399, 0x003333CC, 0x003333FF, 0x00336633, 0x00339933,
509 0x0033CC33, 0x0033FF33, 0x00663333, 0x00993333, 0x00CC3333, 0x00FF3333,
510 0x00003366, 0x00336600, 0x00660033, 0x00006633, 0x00330066, 0x00663300,
511 0x00336699, 0x00669933, 0x00993366, 0x00339966, 0x00663399, 0x00996633,
512 0x006699CC, 0x0099CC66, 0x00CC6699, 0x0066CC99, 0x009966CC, 0x00CC9966,
513 0x0099CCFF, 0x00CCFF99, 0x00FF99CC, 0x0099FFCC, 0x00CC99FF, 0x00FFCC99,
514 0x00111111, 0x00222222, 0x00444444, 0x00555555, 0x00AAAAAA, 0x00BBBBBB,
515 0x00DDDDDD, 0x00EEEEEE
516 };
517
generate_default_palette(Palette * palette)518 static int generate_default_palette(Palette * palette)
519 {
520 memcpy(palette->colors, default_screen_video_v2_palette,
521 sizeof(default_screen_video_v2_palette));
522
523 return update_palette_index(palette);
524 }
525
generate_optimum_palette(Palette * palette,const uint8_t * image,int width,int height,int stride)526 static int generate_optimum_palette(Palette * palette, const uint8_t * image,
527 int width, int height, int stride)
528 {
529 //this isn't implemented yet! Default palette only!
530 return -1;
531 }
532
encode_15_7_sl(Palette * palette,uint8_t * dest,const uint8_t * src,int width,int dist)533 static inline int encode_15_7_sl(Palette * palette, uint8_t * dest,
534 const uint8_t * src, int width, int dist)
535 {
536 int len = 0, x;
537 for (x = 0; x < width; x++) {
538 len += write_pixel_15_7(palette, dest + len, src + 3 * x, dist);
539 }
540 return len;
541 }
542
encode_15_7(Palette * palette,Block * b,const uint8_t * src,int stride,int dist)543 static int encode_15_7(Palette * palette, Block * b, const uint8_t * src,
544 int stride, int dist)
545 {
546 int i;
547 uint8_t *ptr = b->enc;
548 for (i = 0; i < b->start; i++)
549 ptr += encode_15_7_sl(palette, ptr, src + i * stride, b->width, dist);
550 b->sl_begin = ptr;
551 for (; i < b->start + b->len; i++)
552 ptr += encode_15_7_sl(palette, ptr, src + i * stride, b->width, dist);
553 b->sl_end = ptr;
554 for (; i < b->height; i++)
555 ptr += encode_15_7_sl(palette, ptr, src + i * stride, b->width, dist);
556 b->enc_size = ptr - b->enc;
557 return b->enc_size;
558 }
559
encode_block(FlashSV2Context * s,Palette * palette,Block * b,Block * prev,const uint8_t * src,int stride,int comp,int dist,int keyframe)560 static int encode_block(FlashSV2Context *s, Palette * palette, Block * b,
561 Block * prev, const uint8_t * src, int stride, int comp,
562 int dist, int keyframe)
563 {
564 unsigned buf_size = b->width * b->height * 6;
565 uint8_t *buf = s->blockbuffer;
566 int res;
567
568 if (b->flags & COLORSPACE_15_7) {
569 encode_15_7(palette, b, src, stride, dist);
570 } else {
571 encode_bgr(b, src, stride);
572 }
573
574 if (b->len > 0) {
575 b->data_size = buf_size;
576 res = encode_zlib(b, b->data, &b->data_size, comp);
577 if (res)
578 return res;
579
580 if (!keyframe) {
581 res = encode_zlibprime(b, prev, buf, &buf_size, comp);
582 if (res)
583 return res;
584
585 if (buf_size < b->data_size) {
586 b->data_size = buf_size;
587 memcpy(b->data, buf, buf_size);
588 b->flags |= ZLIB_PRIME_COMPRESS_PREVIOUS;
589 }
590 }
591 } else {
592 b->data_size = 0;
593 }
594 return 0;
595 }
596
compare_sl(FlashSV2Context * s,Block * b,const uint8_t * src,uint8_t * frame,uint8_t * key,int y,int keyframe)597 static int compare_sl(FlashSV2Context * s, Block * b, const uint8_t * src,
598 uint8_t * frame, uint8_t * key, int y, int keyframe)
599 {
600 if (memcmp(src, frame, b->width * 3) != 0) {
601 b->dirty = 1;
602 memcpy(frame, src, b->width * 3);
603 #ifndef FLASHSV2_DUMB
604 s->diff_lines++;
605 #endif
606 }
607 if (memcmp(src, key, b->width * 3) != 0) {
608 if (b->len == 0)
609 b->start = y;
610 b->len = y + 1 - b->start;
611 }
612 return 0;
613 }
614
mark_all_blocks(FlashSV2Context * s,const uint8_t * src,int stride,int keyframe)615 static int mark_all_blocks(FlashSV2Context * s, const uint8_t * src, int stride,
616 int keyframe)
617 {
618 int sl, rsl, col, pos, possl;
619 Block *b;
620 for (sl = s->image_height - 1; sl >= 0; sl--) {
621 for (col = 0; col < s->cols; col++) {
622 rsl = s->image_height - sl - 1;
623 b = s->frame_blocks + col + rsl / s->block_height * s->cols;
624 possl = stride * sl + col * s->block_width * 3;
625 pos = s->image_width * rsl * 3 + col * s->block_width * 3;
626 compare_sl(s, b, src + possl, s->current_frame + pos,
627 s->key_frame + pos, rsl % s->block_height, keyframe);
628 }
629 }
630 #ifndef FLASHSV2_DUMB
631 s->tot_lines += s->image_height * s->cols;
632 #endif
633 return 0;
634 }
635
encode_all_blocks(FlashSV2Context * s,int keyframe)636 static int encode_all_blocks(FlashSV2Context * s, int keyframe)
637 {
638 int row, col, res;
639 uint8_t *data;
640 Block *b, *prev;
641 for (row = 0; row < s->rows; row++) {
642 for (col = 0; col < s->cols; col++) {
643 b = s->frame_blocks + (row * s->cols + col);
644 prev = s->key_blocks + (row * s->cols + col);
645 b->flags = s->use15_7 ? COLORSPACE_15_7 : 0;
646 if (keyframe) {
647 b->start = 0;
648 b->len = b->height;
649 } else if (!b->dirty) {
650 b->start = 0;
651 b->len = 0;
652 b->data_size = 0;
653 continue;
654 } else if (b->start != 0 || b->len != b->height) {
655 b->flags |= HAS_DIFF_BLOCKS;
656 }
657 data = s->current_frame + s->image_width * 3 * s->block_height * row + s->block_width * col * 3;
658 res = encode_block(s, &s->palette, b, prev, data, s->image_width * 3, s->comp, s->dist, keyframe);
659 #ifndef FLASHSV2_DUMB
660 if (b->dirty)
661 s->diff_blocks++;
662 s->comp_size += b->data_size;
663 s->uncomp_size += b->enc_size;
664 #endif
665 if (res)
666 return res;
667 }
668 }
669 #ifndef FLASHSV2_DUMB
670 s->raw_size += s->image_width * s->image_height * 3;
671 s->tot_blocks += s->rows * s->cols;
672 #endif
673 return 0;
674 }
675
write_all_blocks(FlashSV2Context * s,uint8_t * buf,int buf_size)676 static int write_all_blocks(FlashSV2Context * s, uint8_t * buf,
677 int buf_size)
678 {
679 int row, col, buf_pos = 0, len;
680 Block *b;
681 for (row = 0; row < s->rows; row++) {
682 for (col = 0; col < s->cols; col++) {
683 b = s->frame_blocks + row * s->cols + col;
684 len = write_block(b, buf + buf_pos, buf_size - buf_pos);
685 b->start = b->len = b->dirty = 0;
686 if (len < 0)
687 return len;
688 buf_pos += len;
689 }
690 }
691 return buf_pos;
692 }
693
write_bitstream(FlashSV2Context * s,const uint8_t * src,int stride,uint8_t * buf,int buf_size,int keyframe)694 static int write_bitstream(FlashSV2Context * s, const uint8_t * src, int stride,
695 uint8_t * buf, int buf_size, int keyframe)
696 {
697 int buf_pos, res;
698
699 res = mark_all_blocks(s, src, stride, keyframe);
700 if (res)
701 return res;
702 res = encode_all_blocks(s, keyframe);
703 if (res)
704 return res;
705
706 res = write_header(s, buf, buf_size);
707 if (res < 0) {
708 return res;
709 } else {
710 buf_pos = res;
711 }
712 res = write_all_blocks(s, buf + buf_pos, buf_size - buf_pos);
713 if (res < 0)
714 return res;
715 buf_pos += res;
716 #ifndef FLASHSV2_DUMB
717 s->total_bits += ((double) buf_pos) * 8.0;
718 #endif
719
720 return buf_pos;
721 }
722
recommend_keyframe(FlashSV2Context * s,int * keyframe)723 static void recommend_keyframe(FlashSV2Context * s, int *keyframe)
724 {
725 #ifndef FLASHSV2_DUMB
726 double block_ratio, line_ratio, enc_ratio, comp_ratio, data_ratio;
727 if (s->avctx->gop_size > 0) {
728 block_ratio = s->diff_blocks / s->tot_blocks;
729 line_ratio = s->diff_lines / s->tot_lines;
730 enc_ratio = s->uncomp_size / s->raw_size;
731 comp_ratio = s->comp_size / s->uncomp_size;
732 data_ratio = s->comp_size / s->raw_size;
733
734 if ((block_ratio >= 0.5 && line_ratio / block_ratio <= 0.5) || line_ratio >= 0.95) {
735 *keyframe = 1;
736 return;
737 }
738 }
739 #else
740 return;
741 #endif
742 }
743
744 #ifndef FLASHSV2_DUMB
745 static const double block_size_fraction = 1.0 / 300;
746 static const double use15_7_threshold = 8192;
747 static const double color15_7_factor = 100;
748 #endif
optimum_block_width(FlashSV2Context * s)749 static int optimum_block_width(FlashSV2Context * s)
750 {
751 #ifndef FLASHSV2_DUMB
752 double save = (1-pow(s->diff_lines/s->diff_blocks/s->block_height, 0.5)) * s->comp_size/s->tot_blocks;
753 double width = block_size_fraction * sqrt(0.5 * save * s->rows * s->cols) * s->image_width;
754 int pwidth = ((int) width);
755 return FFCLIP(pwidth & ~15, 256, 16);
756 #else
757 return 64;
758 #endif
759 }
760
optimum_block_height(FlashSV2Context * s)761 static int optimum_block_height(FlashSV2Context * s)
762 {
763 #ifndef FLASHSV2_DUMB
764 double save = (1-pow(s->diff_lines/s->diff_blocks/s->block_height, 0.5)) * s->comp_size/s->tot_blocks;
765 double height = block_size_fraction * sqrt(0.5 * save * s->rows * s->cols) * s->image_height;
766 int pheight = ((int) height);
767 return FFCLIP(pheight & ~15, 256, 16);
768 #else
769 return 64;
770 #endif
771 }
772
optimum_use15_7(FlashSV2Context * s)773 static int optimum_use15_7(FlashSV2Context * s)
774 {
775 #ifndef FLASHSV2_DUMB
776 double ideal = ((double)(s->avctx->bit_rate * s->avctx->time_base.den * s->avctx->ticks_per_frame)) /
777 ((double) s->avctx->time_base.num) * s->avctx->frame_number;
778 if (ideal + use15_7_threshold < s->total_bits) {
779 return 1;
780 } else {
781 return 0;
782 }
783 #else
784 return s->avctx->global_quality == 0;
785 #endif
786 }
787
optimum_dist(FlashSV2Context * s)788 static int optimum_dist(FlashSV2Context * s)
789 {
790 #ifndef FLASHSV2_DUMB
791 double ideal =
792 s->avctx->bit_rate * s->avctx->time_base.den *
793 s->avctx->ticks_per_frame;
794 int dist = pow((s->total_bits / ideal) * color15_7_factor, 3);
795 av_log(s->avctx, AV_LOG_DEBUG, "dist: %d\n", dist);
796 return dist;
797 #else
798 return 15;
799 #endif
800 }
801
802
reconfigure_at_keyframe(FlashSV2Context * s,const uint8_t * image,int stride)803 static int reconfigure_at_keyframe(FlashSV2Context * s, const uint8_t * image,
804 int stride)
805 {
806 int update_palette = 0;
807 int res;
808 int block_width = optimum_block_width (s);
809 int block_height = optimum_block_height(s);
810
811 if (block_width != s->block_width || block_height != s->block_height) {
812 res = update_block_dimensions(s, block_width, block_height);
813 if (res < 0)
814 return res;
815 }
816
817 s->use15_7 = optimum_use15_7(s);
818 if (s->use15_7) {
819 if ((s->use_custom_palette && s->palette_type != 1) || update_palette) {
820 res = generate_optimum_palette(&s->palette, image, s->image_width, s->image_height, stride);
821 if (res)
822 return res;
823 s->palette_type = 1;
824 av_log(s->avctx, AV_LOG_DEBUG, "Generated optimum palette\n");
825 } else if (!s->use_custom_palette && s->palette_type != 0) {
826 res = generate_default_palette(&s->palette);
827 if (res)
828 return res;
829 s->palette_type = 0;
830 av_log(s->avctx, AV_LOG_DEBUG, "Generated default palette\n");
831 }
832 }
833
834
835 reset_stats(s);
836
837 return 0;
838 }
839
flashsv2_encode_frame(AVCodecContext * avctx,AVPacket * pkt,const AVFrame * p,int * got_packet)840 static int flashsv2_encode_frame(AVCodecContext *avctx, AVPacket *pkt,
841 const AVFrame *p, int *got_packet)
842 {
843 FlashSV2Context *const s = avctx->priv_data;
844 int res;
845 int keyframe = 0;
846
847 if ((res = ff_alloc_packet2(avctx, pkt, s->frame_size + AV_INPUT_BUFFER_MIN_SIZE, 0)) < 0)
848 return res;
849
850 /* First frame needs to be a keyframe */
851 if (avctx->frame_number == 0)
852 keyframe = 1;
853
854 /* Check the placement of keyframes */
855 if (avctx->gop_size > 0) {
856 if (avctx->frame_number >= s->last_key_frame + avctx->gop_size)
857 keyframe = 1;
858 }
859
860 if (!keyframe
861 && avctx->frame_number > s->last_key_frame + avctx->keyint_min) {
862 recommend_keyframe(s, &keyframe);
863 if (keyframe)
864 av_log(avctx, AV_LOG_DEBUG, "Recommending key frame at frame %d\n", avctx->frame_number);
865 }
866
867 if (keyframe) {
868 res = reconfigure_at_keyframe(s, p->data[0], p->linesize[0]);
869 if (res)
870 return res;
871 }
872
873 if (s->use15_7)
874 s->dist = optimum_dist(s);
875
876 res = write_bitstream(s, p->data[0], p->linesize[0], pkt->data, pkt->size, keyframe);
877
878 if (keyframe) {
879 new_key_frame(s);
880 s->last_key_frame = avctx->frame_number;
881 pkt->flags |= AV_PKT_FLAG_KEY;
882 av_log(avctx, AV_LOG_DEBUG, "Inserting key frame at frame %d\n", avctx->frame_number);
883 }
884
885 pkt->size = res;
886 *got_packet = 1;
887
888 return 0;
889 }
890
flashsv2_encode_end(AVCodecContext * avctx)891 static av_cold int flashsv2_encode_end(AVCodecContext * avctx)
892 {
893 FlashSV2Context *s = avctx->priv_data;
894
895 cleanup(s);
896
897 return 0;
898 }
899
900 AVCodec ff_flashsv2_encoder = {
901 .name = "flashsv2",
902 .long_name = NULL_IF_CONFIG_SMALL("Flash Screen Video Version 2"),
903 .type = AVMEDIA_TYPE_VIDEO,
904 .id = AV_CODEC_ID_FLASHSV2,
905 .priv_data_size = sizeof(FlashSV2Context),
906 .init = flashsv2_encode_init,
907 .encode2 = flashsv2_encode_frame,
908 .close = flashsv2_encode_end,
909 .pix_fmts = (const enum AVPixelFormat[]){ AV_PIX_FMT_BGR24, AV_PIX_FMT_NONE },
910 .caps_internal = FF_CODEC_CAP_INIT_CLEANUP,
911 };
912