cairo-templates.h revision 49d5f79a53a49f94f221ef19257bd2577a384b68
/**
* @file
* @brief Cairo software blending templates
*//*
* Authors:
* Krzysztof KosiĆski <tweenk.pl@gmail.com>
*
* Copyright (C) 2010 Authors
* Released under GNU GPL, read the file 'COPYING' for more information
*/
#ifdef HAVE_OPENMP
#include <omp.h>
#include "preferences.h"
#endif
#include <algorithm>
#include <cairo.h>
#include <glib.h>
/**
* @brief Blend two surfaces using the supplied functor.
* This template blends two Cairo image surfaces using a blending functor that takes
* two 32-bit ARGB pixel values and returns a modified 32-bit pixel value.
* Differences in input surface formats are handled transparently. In future, this template
* will also handle software fallback for GL surfaces. */
void ink_cairo_surface_blend(cairo_surface_t *in1, cairo_surface_t *in2, cairo_surface_t *out, Blend blend)
{
// ASSUMPTIONS
// 1. Cairo ARGB32 surface strides are always divisible by 4
// 2. We can only receive CAIRO_FORMAT_ARGB32 or CAIRO_FORMAT_A8 surfaces
// 3. Both surfaces are of the same size
// 4. Output surface is ARGB32 if at least one input is ARGB32
int w = cairo_image_surface_get_width(in2);
int h = cairo_image_surface_get_height(in2);
// Check whether we can loop over pixels without taking stride into account.
bool fast_path = true;
int limit = w * h;
// NOTE
// OpenMP probably doesn't help much here.
// It would be better to render more than 1 tile at a time.
#if HAVE_OPENMP
int num_threads = prefs->getIntLimited("/options/threading/numthreads", omp_get_num_procs(), 1, 256);
#endif
// The number of code paths here is evil.
if (bpp1 == 4) {
if (bpp2 == 4) {
if (fast_path) {
#if HAVE_OPENMP
#endif
for (int i = 0; i < limit; ++i) {
}
} else {
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
}
}
}
} else {
// bpp2 == 1
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
in2_px <<= 24;
}
}
}
} else {
if (bpp2 == 4) {
// bpp1 == 1
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
in1_px <<= 24;
}
}
} else {
// bpp1 == 1 && bpp2 == 1
if (fast_path) {
#if HAVE_OPENMP
#endif
for (int i = 0; i < limit; ++i) {
}
} else {
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
}
}
}
}
}
}
{
// ASSUMPTIONS
// 1. Cairo ARGB32 surface strides are always divisible by 4
// 2. We can only receive CAIRO_FORMAT_ARGB32 or CAIRO_FORMAT_A8 surfaces
// 3. Surfaces have the same dimensions
// 4. Output surface is A8 if input is A8
int w = cairo_image_surface_get_width(in);
int h = cairo_image_surface_get_height(in);
int limit = w * h;
// Check whether we can loop over pixels without taking stride into account.
bool fast_path = true;
#if HAVE_OPENMP
int num_threads = prefs->getIntLimited("/options/threading/numthreads", omp_get_num_procs(), 1, 256);
#endif
if (bppin == 4) {
if (bppout == 4) {
// bppin == 4, bppout == 4
if (fast_path) {
#if HAVE_OPENMP
#endif
for (int i = 0; i < limit; ++i) {
}
} else {
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
}
}
}
} else {
// bppin == 4, bppout == 1
// we use this path with COLORMATRIX_LUMINANCETOALPHA
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
}
}
}
} else {
// bppin == 1, bppout == 1
// Note: there is no path for bppin == 1, bppout == 4 because it is useless
if (fast_path) {
#if HAVE_OPENMP
#endif
for (int i = 0; i < limit; ++i) {
}
} else {
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
}
}
}
}
}
{
// ASSUMPTIONS
// 1. Cairo ARGB32 surface strides are always divisible by 4
// 2. We can only receive CAIRO_FORMAT_ARGB32 or CAIRO_FORMAT_A8 surfaces
int w = cairo_image_surface_get_width(out);
int h = cairo_image_surface_get_height(out);
int limit = w * h;
// NOTE: fast path is not used, because we would need 2 divisions to get pixel indices
#if HAVE_OPENMP
int num_threads = prefs->getIntLimited("/options/threading/numthreads", omp_get_num_procs(), 1, 256);
#endif
if (bppout == 4) {
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
++out_p;
}
}
} else {
// bppout == 1
#if HAVE_OPENMP
#endif
for (int i = 0; i < h; ++i) {
for (int j = 0; j < w; ++j) {
++out_p;
}
}
}
}
// simple pixel accessor for image surface that handles different edge wrapping modes
typedef PixelAccessor self;
enum EdgeMode {
};
: _surface(s)
, _edge_mode(e)
{}
// This is a lot of ifs for a single pixel access. However, branch prediction
// should help us a lot, as the result of ifs is always the same for a single image.
switch (_edge_mode) {
case EDGE_PAD:
break;
case EDGE_WRAP:
break;
case EDGE_ZERO:
default:
return 0;
break;
}
if (_alpha) {
} else {
return *px;
}
}
bool _alpha;
};
// Some helpers for pixel manipulation
G_GNUC_CONST inline gint32
return v;
}
#define EXTRACT_ARGB32(px,a,r,g,b) \
guint32 a, r, g, b; \
b = (px & 0x000000ff);
#define ASSEMBLE_ARGB32(px,a,r,g,b) \
#endif
/*
Local Variables:
mode:c++
c-file-style:"stroustrup"
c-file-offsets:((innamespace . 0)(inline-open . 0)(case-label . +))
indent-tabs-mode:nil
fill-column:99
End:
*/
// vim: filetype=cpp:expandtab:shiftwidth=4:tabstop=8:softtabstop=4:encoding=utf-8:textwidth=99 :