Browse code

vf_overlay: add support to alpha pre-multiplication in the RGBA path

Based on the work of Mark Himsley <mark@mdsh.com>.

See thread:
Subject: [FFmpeg-devel] libavfilter: extending overlay filter
Date: Sun, 13 Mar 2011 14:18:42 +0000

Stefano Sabatini authored on 2011/10/29 07:29:25
Showing 1 changed files
... ...
@@ -360,18 +360,35 @@ static void blend_slice(AVFilterContext *ctx,
360 360
         const int dr = over->main_rgba_map[R];
361 361
         const int dg = over->main_rgba_map[G];
362 362
         const int db = over->main_rgba_map[B];
363
+        const int da = over->main_rgba_map[A];
363 364
         const int dstep = over->main_pix_step[0];
364 365
         const int sr = over->overlay_rgba_map[R];
365 366
         const int sg = over->overlay_rgba_map[G];
366 367
         const int sb = over->overlay_rgba_map[B];
367 368
         const int sa = over->overlay_rgba_map[A];
368 369
         const int sstep = over->overlay_pix_step[0];
370
+        const int main_has_alpha = over->main_has_alpha;
369 371
         if (slice_y > y)
370 372
             sp += (slice_y - y) * src->linesize[0];
371 373
         for (i = 0; i < height; i++) {
372 374
             uint8_t *d = dp, *s = sp;
373 375
             for (j = 0; j < width; j++) {
374 376
                 alpha = s[sa];
377
+
378
+                // if the main channel has an alpha channel, alpha has to be calculated
379
+                // to create an un-premultiplied (straight) alpha value
380
+                if (main_has_alpha && alpha != 0 && alpha != 255) {
381
+                    // apply the general equation:
382
+                    // alpha = alpha_overlay / ( (alpha_main + alpha_overlay) - (alpha_main * alpha_overlay) )
383
+                    alpha =
384
+                        // the next line is a faster version of: 255 * 255 * alpha
385
+                        ( (alpha << 16) - (alpha << 9) + alpha )
386
+                        /
387
+                        // the next line is a faster version of: 255 * (alpha + d[da])
388
+                        ( ((alpha + d[da]) << 8 ) - (alpha + d[da])
389
+                          - d[da] * alpha );
390
+                }
391
+
375 392
                 switch (alpha) {
376 393
                 case 0:
377 394
                     break;
... ...
@@ -387,6 +404,18 @@ static void blend_slice(AVFilterContext *ctx,
387 387
                     d[dg] = FAST_DIV255(d[dg] * (255 - alpha) + s[sg] * alpha);
388 388
                     d[db] = FAST_DIV255(d[db] * (255 - alpha) + s[sb] * alpha);
389 389
                 }
390
+                if (main_has_alpha) {
391
+                    switch (alpha) {
392
+                    case 0:
393
+                        break;
394
+                    case 255:
395
+                        d[da] = s[sa];
396
+                        break;
397
+                    default:
398
+                        // apply alpha compositing: main_alpha += (1-main_alpha) * overlay_alpha
399
+                        d[da] += FAST_DIV255((255 - d[da]) * s[sa]);
400
+                    }
401
+                }
390 402
                 d += dstep;
391 403
                 s += sstep;
392 404
             }