efl/legacy/evas/src/modules/engines/software_x11/evas_xcb_outbuf.c

1145 lines
36 KiB
C
Raw Normal View History

#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include <xcb/shm.h>
#include <xcb/xcb_image.h>
#include <pixman.h>
#include <sys/time.h>
#include <sys/utsname.h>
#include "evas_common.h"
#include "evas_macros.h"
#include "evas_xcb_outbuf.h"
#include "evas_xcb_buffer.h"
#include "evas_xcb_color.h"
typedef struct _Outbuf_Region Outbuf_Region;
struct _Outbuf_Region
{
Xcb_Output_Buffer *xcbob;
Xcb_Output_Buffer *mxcbob;
int x;
int y;
int w;
int h;
};
static Eina_List *shmpool = NULL;
static int shmsize = 0;
static int shmmemlimit = 10 * 1024 * 1024;
static int shmcountlimit = 32;
static Xcb_Output_Buffer *
_find_xcbob(xcb_connection_t *conn, int depth, int w, int h, int shm, void *data)
{
Eina_List *l;
Eina_List *xl = NULL;
Xcb_Output_Buffer *xcbob = NULL;
Xcb_Output_Buffer *xcbob2;
int fitness = 0x7fffffff;
int sz;
int lbytes;
int bpp;
if (!shm)
return evas_software_xcb_x_output_buffer_new(conn, depth, w, h, shm, data);
if (depth > 1)
{
bpp = depth / 8;
if (bpp == 3) bpp = 4;
lbytes = (((w * bpp) + 3) / 4) * 4;
}
else
lbytes = ((w + 31) / 32) * 4;
sz = lbytes * h;
EINA_LIST_FOREACH(shmpool, l, xcbob2)
{
int szdif;
if ((xcbob2->image->depth != depth) ||
(xcbob2->connection != conn))
continue;
szdif = xcbob2->psize - sz;
if (szdif < 0) continue;
if (szdif == 0)
{
xcbob = xcbob2;
xl = l;
goto have_xcbob;
}
if (szdif < fitness)
{
fitness = szdif;
xcbob = xcbob2;
xl = l;
}
}
if ((fitness > (100 * 100)) || (!xcbob))
return evas_software_xcb_x_output_buffer_new(conn, depth, w, h, shm, data);
have_xcbob:
shmpool = eina_list_remove_list(shmpool, xl);
xcbob->w = w;
xcbob->h = h;
xcbob->bpl = lbytes;
xcbob->image->width = xcbob->w;
xcbob->image->height = xcbob->h;
xcbob->image->stride = xcbob->bpl;
shmsize -= xcbob->psize * (xcbob->image->depth / 8);
return xcbob;
}
static void
_unfind_xcbob(Xcb_Output_Buffer *xcbob, int sync)
{
if (xcbob->shm_info)
{
shmpool = eina_list_prepend(shmpool, xcbob);
shmsize += xcbob->psize * xcbob->image->depth / 8;
while ((shmsize > (shmmemlimit)) ||
(eina_list_count(shmpool) > shmcountlimit))
{
Eina_List *xl;
xl = eina_list_last(shmpool);
if (!xl)
{
shmsize = 0;
break;
}
xcbob = xl->data;
shmpool = eina_list_remove_list(shmpool, xl);
evas_software_xcb_x_output_buffer_free(xcbob, sync);
}
}
else
evas_software_xcb_x_output_buffer_free(xcbob, sync);
}
static void
_clear_xcbob(int sync)
{
while (shmpool)
{
Xcb_Output_Buffer *xcbob;
xcbob = shmpool->data;
shmpool = eina_list_remove_list(shmpool, shmpool);
evas_software_xcb_x_output_buffer_free(xcbob, sync);
}
shmsize = 0;
}
void
evas_software_xcb_outbuf_init(void)
{
}
void
evas_software_xcb_outbuf_free(Outbuf * buf)
{
while (buf->priv.pending_writes)
{
RGBA_Image *im;
Outbuf_Region *obr;
im = buf->priv.pending_writes->data;
buf->priv.pending_writes = eina_list_remove_list(buf->priv.pending_writes, buf->priv.pending_writes);
obr = im->extended_info;
evas_cache_image_drop(&im->cache_entry);
if (obr->xcbob) _unfind_xcbob(obr->xcbob, 0);
if (obr->mxcbob) _unfind_xcbob(obr->mxcbob, 0);
free(obr);
}
evas_software_xcb_outbuf_idle_flush(buf);
evas_software_xcb_outbuf_flush(buf);
if (buf->priv.x11.xcb.gc)
xcb_free_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gc);
if (buf->priv.x11.xcb.gcm)
xcb_free_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gcm);
if (buf->priv.pal)
evas_software_xcb_x_color_deallocate(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.cmap,
buf->priv.x11.xcb.vis,
buf->priv.pal);
free(buf);
_clear_xcbob(0);
}
void
evas_software_xcb_outbuf_rotation_set(Outbuf *buf, int rot)
{
buf->rot = rot;
}
Outbuf *
evas_software_xcb_outbuf_setup_x(int w,
int h,
int rot,
Outbuf_Depth depth,
xcb_connection_t *conn,
xcb_screen_t *screen,
xcb_drawable_t draw,
xcb_visualtype_t *vis,
xcb_colormap_t cmap,
int x_depth,
int grayscale,
int max_colors,
xcb_drawable_t mask,
int shape_dither,
int destination_alpha)
{
Outbuf *buf;
buf = calloc(1, sizeof(Outbuf));
if (!buf)
return NULL;
buf->w = w;
buf->h = h;
buf->depth = depth;
buf->rot = rot;
buf->priv.x11.xcb.conn = conn;
buf->priv.x11.xcb.screen = screen;
buf->priv.x11.xcb.vis = vis;
buf->priv.x11.xcb.cmap = cmap;
buf->priv.x11.xcb.depth = x_depth;
buf->priv.mask_dither = shape_dither;
buf->priv.destination_alpha = destination_alpha;
{
Gfx_Func_Convert conv_func;
Xcb_Output_Buffer *xcbob;
buf->priv.x11.xcb.shm = evas_software_xcb_x_can_do_shm(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.screen);
xcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
1, 1,
buf->priv.x11.xcb.shm, NULL);
conv_func = NULL;
if (xcbob)
{
#ifdef WORDS_BIGENDIAN
if (evas_software_xcb_x_output_buffer_byte_order(xcbob) == XCB_IMAGE_ORDER_LSB_FIRST)
buf->priv.x11.xcb.swap = 1;
if (evas_software_xcb_x_output_buffer_bit_order(xcbob) == XCB_IMAGE_ORDER_MSB_FIRST)
buf->priv.x11.xcb.bit_swap = 1;
#else
if (evas_software_xcb_x_output_buffer_byte_order(xcbob) == XCB_IMAGE_ORDER_MSB_FIRST)
buf->priv.x11.xcb.swap = 1;
if (evas_software_xcb_x_output_buffer_bit_order(xcbob) == XCB_IMAGE_ORDER_MSB_FIRST)
buf->priv.x11.xcb.bit_swap = 1;
#endif
if (((vis->_class == XCB_VISUAL_CLASS_TRUE_COLOR) ||
(vis->_class == XCB_VISUAL_CLASS_DIRECT_COLOR)) &&
(x_depth > 8))
{
buf->priv.mask.r = (DATA32) vis->red_mask;
buf->priv.mask.g = (DATA32) vis->green_mask;
buf->priv.mask.b = (DATA32) vis->blue_mask;
if (buf->priv.x11.xcb.swap)
{
SWAP32(buf->priv.mask.r);
SWAP32(buf->priv.mask.g);
SWAP32(buf->priv.mask.b);
}
}
else if ((vis->_class == XCB_VISUAL_CLASS_STATIC_GRAY) ||
(vis->_class == XCB_VISUAL_CLASS_GRAY_SCALE) ||
(vis->_class == XCB_VISUAL_CLASS_STATIC_COLOR) ||
(vis->_class == XCB_VISUAL_CLASS_PSEUDO_COLOR) ||
(x_depth <= 8))
{
Convert_Pal_Mode pm = PAL_MODE_RGB332;
if ((vis->_class == XCB_VISUAL_CLASS_GRAY_SCALE) ||
(vis->_class == XCB_VISUAL_CLASS_STATIC_GRAY))
grayscale = 1;
if (grayscale)
{
if (max_colors >= 256)
pm = PAL_MODE_GRAY256;
else if (max_colors >= 64)
pm = PAL_MODE_GRAY64;
else if (max_colors >= 16)
pm = PAL_MODE_GRAY16;
else if (max_colors >= 4)
pm = PAL_MODE_GRAY4;
else
pm = PAL_MODE_MONO;
}
else
{
if (max_colors >= 256)
pm = PAL_MODE_RGB332;
else if (max_colors >= 216)
pm = PAL_MODE_RGB666;
else if (max_colors >= 128)
pm = PAL_MODE_RGB232;
else if (max_colors >= 64)
pm = PAL_MODE_RGB222;
else if (max_colors >= 32)
pm = PAL_MODE_RGB221;
else if (max_colors >= 16)
pm = PAL_MODE_RGB121;
else if (max_colors >= 8)
pm = PAL_MODE_RGB111;
else if (max_colors >= 4)
pm = PAL_MODE_GRAY4;
else
pm = PAL_MODE_MONO;
}
/* FIXME: only alloc once per display+cmap */
buf->priv.pal =
evas_software_xcb_x_color_allocate(conn,
cmap,
vis,
pm);
if (!buf->priv.pal)
{
free(buf);
return NULL;
}
}
if (buf->priv.pal)
{
if (buf->rot == 0 || buf->rot == 180)
conv_func = evas_common_convert_func_get(0,
buf->w,
buf->h,
evas_software_xcb_x_output_buffer_depth (xcbob),
buf->priv.mask.r,
buf->priv.mask.g,
buf->priv.mask.b,
buf->priv.pal->colors,
buf->rot);
else if (buf->rot == 90 || buf->rot == 270)
conv_func = evas_common_convert_func_get(0,
buf->h,
buf->w,
evas_software_xcb_x_output_buffer_depth (xcbob),
buf->priv.mask.r,
buf->priv.mask.g,
buf->priv.mask.b,
buf->priv.pal->colors,
buf->rot);
}
else
{
if (buf->rot == 0 || buf->rot == 180)
conv_func = evas_common_convert_func_get(0,
buf->w,
buf->h,
evas_software_xcb_x_output_buffer_depth(xcbob),
buf->priv.mask.r,
buf->priv.mask.g,
buf->priv.mask.b,
PAL_MODE_NONE,
buf->rot);
else if (buf->rot == 90 || buf->rot == 270)
conv_func = evas_common_convert_func_get(0,
buf->h,
buf->w,
evas_software_xcb_x_output_buffer_depth(xcbob),
buf->priv.mask.r,
buf->priv.mask.g,
buf->priv.mask.b,
PAL_MODE_NONE,
buf->rot);
}
evas_software_xcb_x_output_buffer_free(xcbob, 1);
if (!conv_func)
{
ERR("XCB Engine"
" {"
" At depth %i:"
" RGB format mask: %08x, %08x, %08x"
" Palette mode: %i"
" Not supported by and compiled in converters!"
" }",
buf->priv.x11.xcb.depth,
buf->priv.mask.r,
buf->priv.mask.g,
buf->priv.mask.b,
buf->priv.pal ? buf->priv.pal->colors : -1);
}
}
evas_software_xcb_outbuf_drawable_set(buf, draw);
evas_software_xcb_outbuf_mask_set(buf, mask);
}
return buf;
}
RGBA_Image *
evas_software_xcb_outbuf_new_region_for_update(Outbuf *buf,
int x,
int y,
int w,
int h,
int *cx,
int *cy,
int *cw,
int *ch)
{
RGBA_Image *im;
Outbuf_Region *obr;
int bpl = 0;
int use_shm = 1;
int alpha;
if ((buf->onebuf) && (buf->priv.x11.xcb.shm))
{
Eina_Rectangle *rect;
RECTS_CLIP_TO_RECT(x, y, w, h, 0, 0, buf->w, buf->h);
obr = calloc(1, sizeof(Outbuf_Region));
if (!obr) return NULL;
rect = eina_rectangle_new(x, y, w, h);
if (!rect)
{
free(obr);
return NULL;
}
buf->priv.onebuf_regions = eina_list_append(buf->priv.onebuf_regions, rect);
if (buf->priv.onebuf)
{
*cx = x;
*cy = y;
*cw = w;
*ch = h;
if (!buf->priv.synced)
{
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
buf->priv.synced = 1;
}
return buf->priv.onebuf;
}
obr->x = 0;
obr->y = 0;
obr->w = buf->w;
obr->h = buf->h;
*cx = x;
*cy = y;
*cw = w;
*ch = h;
alpha = ((buf->priv.x11.xcb.mask) || (buf->priv.destination_alpha));
use_shm = buf->priv.x11.xcb.shm;
if ((buf->rot == 0) &&
(buf->priv.mask.r == 0xff0000) &&
(buf->priv.mask.g == 0x00ff00) &&
(buf->priv.mask.b == 0x0000ff))
{
obr->xcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
buf->w, buf->h,
use_shm,
NULL);
if (!obr->xcbob)
{
free(obr);
return NULL;
}
im = (RGBA_Image *) evas_cache_image_data(evas_common_image_cache_get(),
buf->w, buf->h,
(DATA32 *)evas_software_xcb_x_output_buffer_data(obr->xcbob, &bpl),
alpha, EVAS_COLORSPACE_ARGB8888);
if (!im)
{
evas_software_xcb_x_output_buffer_free(obr->xcbob, 0);
free(obr);
return NULL;
}
im->extended_info = obr;
if (buf->priv.x11.xcb.mask)
obr->mxcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
1,
buf->w, buf->h,
use_shm,
NULL);
}
else
{
im = (RGBA_Image *) evas_cache_image_empty(evas_common_image_cache_get());
if (!im)
{
free(obr);
return NULL;
}
im->cache_entry.flags.alpha |= alpha ? 1 : 0;
evas_cache_image_surface_alloc(&im->cache_entry, buf->w, buf->h);
im->extended_info = obr;
if ((buf->rot == 0) || (buf->rot == 180))
{
obr->xcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
buf->w, buf->h,
use_shm,
NULL);
if (!obr->xcbob)
{
evas_cache_image_drop(&im->cache_entry);
free(obr);
return NULL;
}
if (buf->priv.x11.xcb.mask)
obr->mxcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
1,
buf->w, buf->h,
use_shm,
NULL);
}
else if ((buf->rot == 90) || (buf->rot == 270))
{
obr->xcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
buf->h, buf->w,
use_shm,
NULL);
if (!obr->xcbob)
{
evas_cache_image_drop(&im->cache_entry);
free(obr);
return NULL;
}
if (buf->priv.x11.xcb.mask)
obr->mxcbob = evas_software_xcb_x_output_buffer_new(buf->priv.x11.xcb.conn,
1,
buf->h, buf->w,
use_shm,
NULL);
}
}
/* FIXME: We should be able to remove this memset, but somewhere in the process
we copy too much to the destination surface and some area are not cleaned before copy. */
if ((alpha) && (im->image.data))
{
/* FIXME: faster memset! */
memset(im->image.data, 0, w * h * sizeof(DATA32));
}
buf->priv.onebuf = im;
return im;
}
obr = calloc(1, sizeof(Outbuf_Region));
if (!obr) return NULL;
obr->x = x;
obr->y = y;
obr->w = w;
obr->h = h;
*cx = 0;
*cy = 0;
*cw = w;
*ch = h;
use_shm = buf->priv.x11.xcb.shm;
/* FIXME: magic - i found if shm regions are smaller than 200x200 its
* faster to use ximages over unix sockets - trial and error
*/
// use_shm = 0; /* 630 -> 1006 fps */
// if ((w * h) < (200 * 200)) use_shm = 0; /* 630 -> 962 fps */
alpha = ((buf->priv.x11.xcb.mask) || (buf->priv.destination_alpha));
if ((buf->rot == 0) &&
(buf->priv.mask.r == 0xff0000) &&
(buf->priv.mask.g == 0x00ff00) &&
(buf->priv.mask.b == 0x0000ff))
{
obr->xcbob = _find_xcbob(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
w, h,
use_shm,
NULL);
if (!obr->xcbob)
{
free(obr);
return NULL;
}
im = (RGBA_Image *)evas_cache_image_data(evas_common_image_cache_get(),
w, h,
(DATA32 *) evas_software_xcb_x_output_buffer_data(obr->xcbob, &bpl),
alpha, EVAS_COLORSPACE_ARGB8888);
if (!im)
{
_unfind_xcbob(obr->xcbob, 0);
free(obr);
return NULL;
}
im->extended_info = obr;
if (buf->priv.x11.xcb.mask)
obr->mxcbob = _find_xcbob(buf->priv.x11.xcb.conn,
1,
w, h,
use_shm,
NULL);
}
else
{
im = (RGBA_Image *)evas_cache_image_empty(evas_common_image_cache_get());
if (!im)
{
free(obr);
return NULL;
}
im->cache_entry.flags.alpha |= alpha ? 1 : 0;
evas_cache_image_surface_alloc(&im->cache_entry, w, h);
im->extended_info = obr;
if ((buf->rot == 0) || (buf->rot == 180))
{
obr->xcbob = _find_xcbob(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
w, h,
use_shm,
NULL);
if (!obr->xcbob)
{
evas_cache_image_drop(&im->cache_entry);
free(obr);
return NULL;
}
if (buf->priv.x11.xcb.mask)
obr->mxcbob = _find_xcbob(buf->priv.x11.xcb.conn,
1,
w, h,
use_shm,
NULL);
}
else if ((buf->rot == 90) || (buf->rot == 270))
{
obr->xcbob = _find_xcbob(buf->priv.x11.xcb.conn,
buf->priv.x11.xcb.depth,
h, w,
use_shm,
NULL);
if (!obr->xcbob)
{
evas_cache_image_drop(&im->cache_entry);
free(obr);
return NULL;
}
if (buf->priv.x11.xcb.mask)
obr->mxcbob = _find_xcbob(buf->priv.x11.xcb.conn,
1,
h, w,
use_shm,
NULL);
}
}
/* FIXME: We should be able to remove this memset, but somewhere in the process
we copy too much to the destination surface and some area are not cleaned before copy. */
if (((buf->priv.x11.xlib.mask) || (buf->priv.destination_alpha)) &&
(im->image.data))
{
/* FIXME: faster memset! */
memset(im->image.data, 0, w * h * sizeof(DATA32));
}
buf->priv.pending_writes = eina_list_append(buf->priv.pending_writes, im);
return im;
}
void
evas_software_xcb_outbuf_free_region_for_update(Outbuf *buf __UNUSED__,
RGBA_Image *update __UNUSED__)
{
/* no need to do anything - they are cleaned up on flush */
}
void
evas_software_xcb_outbuf_flush(Outbuf *buf)
{
Eina_List *l;
RGBA_Image *im;
Outbuf_Region *obr;
if ((buf->priv.onebuf) && (buf->priv.onebuf_regions))
{
pixman_region16_t tmpr;
im = buf->priv.onebuf;
obr = im->extended_info;
pixman_region_init(&tmpr);
while (buf->priv.onebuf_regions)
{
Eina_Rectangle *rect;
rect = buf->priv.onebuf_regions->data;
buf->priv.onebuf_regions = eina_list_remove_list(buf->priv.onebuf_regions, buf->priv.onebuf_regions);
pixman_region_union_rect(&tmpr, &tmpr,
rect->x, rect->y,
rect->w, rect->h);
if (buf->priv.debug)
evas_software_xcb_outbuf_debug_show(buf, buf->priv.x11.xcb.win,
rect->x, rect->y, rect->w, rect->h);
eina_rectangle_free(rect);
}
xcb_set_clip_rectangles(buf->priv.x11.xcb.conn, XCB_CLIP_ORDERING_YX_BANDED,
buf->priv.x11.xcb.gc,
0, 0, pixman_region_n_rects(&tmpr),
(const xcb_rectangle_t *)pixman_region_rectangles(&tmpr, NULL));
if (obr->xcbob)
evas_software_xcb_x_output_buffer_paste(obr->xcbob, buf->priv.x11.xcb.win,
buf->priv.x11.xcb.gc,
0, 0, 0);
if (obr->mxcbob)
{
xcb_set_clip_rectangles(buf->priv.x11.xcb.conn, XCB_CLIP_ORDERING_YX_BANDED,
buf->priv.x11.xcb.gcm,
0, 0, pixman_region_n_rects(&tmpr),
(const xcb_rectangle_t *)pixman_region_rectangles(&tmpr, NULL));
evas_software_xcb_x_output_buffer_paste(obr->mxcbob,
buf->priv.x11.xcb.mask,
buf->priv.x11.xcb.gcm,
0, 0, 0);
}
buf->priv.synced = 0;
}
else
{
#if 1
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
EINA_LIST_FOREACH(buf->priv.pending_writes, l, im)
{
obr = im->extended_info;
if (buf->priv.debug)
evas_software_xcb_outbuf_debug_show(buf,
buf->priv.x11.xcb.win,
obr->x,
obr->y,
obr->w,
obr->h);
if (obr->xcbob)
evas_software_xcb_x_output_buffer_paste(obr->xcbob,
buf->priv.x11.xcb.win,
buf->priv.x11.xcb.gc,
obr->x,
obr->y, 0);
if (obr->mxcbob)
evas_software_xcb_x_output_buffer_paste(obr->mxcbob,
buf->priv.x11.xcb.mask,
buf->priv.x11.xcb.gcm,
obr->x,
obr->y, 0);
}
while (buf->priv.pending_writes)
{
im = buf->priv.pending_writes->data;
buf->priv.pending_writes = eina_list_remove_list(buf->priv.pending_writes,
buf->priv.pending_writes);
obr = im->extended_info;
evas_cache_image_drop(&im->cache_entry);
if (obr->xcbob) _unfind_xcbob(obr->xcbob, 0);
if (obr->mxcbob) _unfind_xcbob(obr->mxcbob, 0);
free(obr);
}
buf->priv.prev_pending_writes = buf->priv.pending_writes;
buf->priv.pending_writes = NULL;
xcb_flush(buf->priv.x11.xcb.conn);
#else
/* XX async push - disable */
/*
EINA_LIST_FOREACH(buf->priv.pending_writes, l, im)
{
obr = im->extended_info;
if (buf->priv.debug)
evas_software_x11_outbuf_debug_show(buf, buf->priv.x11.xcb.win,
obr->x, obr->y, obr->w, obr->h);
evas_software_x11_x_output_buffer_paste(obr->xcbob, buf->priv.x11.xcb.win,
buf->priv.x11.xcb.gc,
obr->x, obr->y, 0);
if (obr->mxcbob)
evas_software_x11_x_output_buffer_paste(obr->mxcbob,
buf->priv.x11.xcb.mask,
buf->priv.x11.xcb.gcm,
obr->x, obr->y, 0);
}
*/
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
while (buf->priv.pending_writes)
{
im = eina_list_data_get(buf->priv.pending_writes);
buf->priv.pending_writes = eina_list_remove_list(buf->priv.pending_writes, buf->priv.pending_writes);
obr = im->extended_info;
evas_cache_image_drop(&im->cache_entry);
if (obr->xcbob) _unfind_xcbob(obr->xcbob, 0);
if (obr->mxcbob) _unfind_xcbob(obr->mxcbob, 0);
free(obr);
evas_cache_image_drop(&im->cache_entry);
}
#endif
}
evas_common_cpu_end_opt();
}
void
evas_software_xcb_outbuf_idle_flush(Outbuf *buf)
{
if (buf->priv.onebuf)
{
RGBA_Image *im;
Outbuf_Region *obr;
im = buf->priv.onebuf;
buf->priv.onebuf = NULL;
obr = im->extended_info;
if (obr->xcbob) evas_software_xcb_x_output_buffer_free(obr->xcbob, 0);
if (obr->mxcbob) evas_software_xcb_x_output_buffer_free(obr->mxcbob, 0);
free(obr);
evas_cache_image_drop(&im->cache_entry);
}
else
{
if (buf->priv.prev_pending_writes)
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
while (buf->priv.prev_pending_writes)
{
RGBA_Image *im;
Outbuf_Region *obr;
im = buf->priv.prev_pending_writes->data;
buf->priv.prev_pending_writes =
eina_list_remove_list(buf->priv.prev_pending_writes,
buf->priv.prev_pending_writes);
obr = im->extended_info;
evas_cache_image_drop(&im->cache_entry);
if (obr->xcbob) _unfind_xcbob(obr->xcbob, 0);
if (obr->mxcbob) _unfind_xcbob(obr->mxcbob, 0);
free(obr);
}
_clear_xcbob(0);
}
}
void
evas_software_xcb_outbuf_push_updated_region(Outbuf *buf,
RGBA_Image *update,
int x,
int y,
int w,
int h)
{
Gfx_Func_Convert conv_func = NULL;
Outbuf_Region *obr;
DATA32 *src_data;
void *data;
int bpl = 0;
int yy;
obr = update->extended_info;
if (buf->priv.pal)
{
if ((buf->rot == 0) || (buf->rot == 180))
conv_func = evas_common_convert_func_get(0, w, h,
evas_software_xcb_x_output_buffer_depth
(obr->xcbob), buf->priv.mask.r,
buf->priv.mask.g, buf->priv.mask.b,
buf->priv.pal->colors, buf->rot);
else if ((buf->rot == 90) || (buf->rot == 270))
conv_func = evas_common_convert_func_get(0, h, w,
evas_software_xcb_x_output_buffer_depth
(obr->xcbob), buf->priv.mask.r,
buf->priv.mask.g, buf->priv.mask.b,
buf->priv.pal->colors, buf->rot);
}
else
{
if ((buf->rot == 0) || (buf->rot == 180))
conv_func = evas_common_convert_func_get(0, w, h,
evas_software_xcb_x_output_buffer_depth
(obr->xcbob), buf->priv.mask.r,
buf->priv.mask.g, buf->priv.mask.b,
PAL_MODE_NONE, buf->rot);
else if ((buf->rot == 90) || (buf->rot == 270))
conv_func = evas_common_convert_func_get(0, h, w,
evas_software_xcb_x_output_buffer_depth
(obr->xcbob), buf->priv.mask.r,
buf->priv.mask.g, buf->priv.mask.b,
PAL_MODE_NONE, buf->rot);
}
if (!conv_func) return;
if (!obr->xcbob) return;
data = evas_software_xcb_x_output_buffer_data(obr->xcbob, &bpl);
if (!data) return;
src_data = update->image.data;
if (!src_data) return;
if (buf->rot == 0)
{
obr->x = x;
obr->y = y;
}
else if (buf->rot == 90)
{
obr->x = y;
obr->y = buf->w - x - w;
}
else if (buf->rot == 180)
{
obr->x = buf->w - x - w;
obr->y = buf->h - y - h;
}
else if (buf->rot == 270)
{
obr->x = buf->h - y - h;
obr->y = x;
}
if ((buf->rot == 0) || (buf->rot == 180))
{
obr->w = w;
obr->h = h;
}
else if ((buf->rot == 90) || (buf->rot == 270))
{
obr->w = h;
obr->h = w;
}
if (buf->priv.pal)
{
if (data != src_data)
conv_func(src_data, data,
0,
bpl /
((evas_software_xcb_x_output_buffer_depth(obr->xcbob) / 8)) - obr->w,
obr->w, obr->h, x, y,
buf->priv.pal->lookup);
}
else
{
if (data != src_data)
conv_func(src_data, data,
0,
bpl /
((evas_software_xcb_x_output_buffer_depth(obr->xcbob) / 8)) - obr->w,
obr->w, obr->h, x, y, NULL);
}
#if 1
#else
/* XX async push */
if (!((buf->priv.onebuf) && (buf->priv.onebuf_regions)))
{
if (buf->priv.debug)
evas_software_xcb_outbuf_debug_show(buf, buf->priv.x11.xcb.win,
obr->x, obr->y, obr->w, obr->h);
if (obr->xcbob)
evas_software_xcb_x_output_buffer_paste(obr->xcbob, buf->priv.x11.xcb.win,
buf->priv.x11.xcb.gc,
obr->x, obr->y, 0);
}
#endif
if (obr->mxcbob)
{
if (buf->rot == 0)
{
for (yy = 0; yy < obr->h; yy++)
evas_software_xcb_x_write_mask_line(buf, obr->mxcbob,
src_data + (yy * obr->w),
obr->w,
yy);
}
else if (buf->rot == 90)
{
for (yy = 0; yy < obr->h; yy++)
evas_software_xcb_x_write_mask_line_vert(buf, obr->mxcbob,
src_data + yy,
h, // h
obr->h - yy - 1, // ym
w); // w
}
else if (buf->rot == 180)
{
for (yy = 0; yy < obr->h; yy++)
{
evas_software_xcb_x_write_mask_line_rev(buf, obr->mxcbob,
src_data + (yy * obr->w),
obr->w,
obr->h - yy - 1);
}
}
else if (buf->rot == 270)
{
for (yy = 0; yy < obr->h; yy++)
evas_software_xcb_x_write_mask_line_vert_rev(buf, obr->mxcbob,
src_data + yy,
h, // h
yy, // ym
w); // w
}
#if 1
#else
/* XX async push */
if (!((buf->priv.onebuf) && (buf->priv.onebuf_regions)))
evas_software_xcb_x_output_buffer_paste(obr->mxcbob,
buf->priv.x11.xcb.mask,
buf->priv.x11.xcb.gcm,
obr->x, obr->y, 0);
#endif
}
#if 1
#else
xcb_flush(buf->priv.x11.xcb.conn);
#endif
}
void
evas_software_xcb_outbuf_reconfigure(Outbuf *buf,
int w,
int h,
int rot,
Outbuf_Depth depth)
{
if ((w == buf->w) &&
(h == buf->h) &&
(rot == buf->rot) &&
(depth == buf->depth)) return;
buf->w = w;
buf->h = h;
buf->rot = rot;
evas_software_xcb_outbuf_idle_flush(buf);
}
int
evas_software_xcb_outbuf_get_width(Outbuf * buf)
{
return buf->w;
}
int
evas_software_xcb_outbuf_get_height(Outbuf * buf)
{
return buf->h;
}
Outbuf_Depth
evas_software_xcb_outbuf_get_depth(Outbuf * buf)
{
return buf->depth;
}
int
evas_software_xcb_outbuf_get_rot(Outbuf * buf)
{
return buf->rot;
}
void
evas_software_xcb_outbuf_drawable_set(Outbuf *buf,
xcb_drawable_t draw)
{
if (buf->priv.x11.xcb.win == draw) return;
if (buf->priv.x11.xcb.gc)
{
xcb_free_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gc);
buf->priv.x11.xcb.gc = 0;
}
buf->priv.x11.xcb.win = draw;
buf->priv.x11.xcb.gc = xcb_generate_id(buf->priv.x11.xcb.conn);
xcb_create_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gc, buf->priv.x11.xcb.win, 0, NULL);
}
void
evas_software_xcb_outbuf_mask_set(Outbuf *buf,
xcb_drawable_t mask)
{
if (buf->priv.x11.xcb.mask == mask) return;
if (buf->priv.x11.xcb.gcm)
{
xcb_free_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gcm);
buf->priv.x11.xcb.gcm = 0;
}
buf->priv.x11.xcb.mask = mask;
if (buf->priv.x11.xcb.mask)
{
buf->priv.x11.xcb.gcm = xcb_generate_id(buf->priv.x11.xcb.conn);
xcb_create_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gcm, buf->priv.x11.xcb.win, 0, NULL);
}
}
void
evas_software_xcb_outbuf_debug_set(Outbuf *buf,
int debug)
{
buf->priv.debug = debug;
}
void
evas_software_xcb_outbuf_debug_show(Outbuf *buf,
xcb_drawable_t draw,
int x,
int y,
int w,
int h)
{
int i;
xcb_screen_t *screen = NULL;
{
xcb_get_geometry_reply_t *geom;
xcb_drawable_t root;
xcb_screen_iterator_t i;
geom = xcb_get_geometry_reply (buf->priv.x11.xcb.conn, xcb_get_geometry_unchecked(buf->priv.x11.xcb.conn, draw), 0);
root = geom->root;
free (geom);
geom = xcb_get_geometry_reply (buf->priv.x11.xcb.conn, xcb_get_geometry_unchecked(buf->priv.x11.xcb.conn, root), 0);
i = xcb_setup_roots_iterator((xcb_setup_t *)xcb_get_setup(buf->priv.x11.xcb.conn));
for (; i.rem; xcb_screen_next(&i))
{
if (i.data->root == geom->root)
{
screen = i.data;
break;
}
}
free (geom);
}
for (i = 0; i < 20; i++)
{
xcb_rectangle_t rect = { x, y, w, h};
uint32_t mask;
uint32_t value[2];
mask = XCB_GC_FOREGROUND | XCB_GC_GRAPHICS_EXPOSURES;
value[0] = screen->black_pixel;
value[1] = XCB_EXPOSURES_NOT_ALLOWED; /* no graphics exposures allowed */
xcb_change_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gc, mask, value);
xcb_poly_fill_rectangle (buf->priv.x11.xcb.conn, draw, buf->priv.x11.xcb.gc, 1, &rect);
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
mask = XCB_GC_FOREGROUND | XCB_GC_GRAPHICS_EXPOSURES;
value[0] = screen->white_pixel;
value[1] = XCB_EXPOSURES_NOT_ALLOWED; /* no graphics exposures allowed */
xcb_change_gc(buf->priv.x11.xcb.conn, buf->priv.x11.xcb.gc, mask, value);
xcb_poly_fill_rectangle (buf->priv.x11.xcb.conn, draw, buf->priv.x11.xcb.gc, 1, &rect);
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
/* we sync */
free(xcb_get_input_focus_reply(buf->priv.x11.xcb.conn, xcb_get_input_focus_unchecked(buf->priv.x11.xcb.conn), NULL));
}
}
Eina_Bool
evas_software_xcb_outbuf_alpha_get (Outbuf *buf)
{
return buf->priv.x11.xcb.mask;
}