evas: reduce usage of __thread directive.

Moved rects caching into draw context to avoid the use of __thread
slot. Draw context are defined per thread anyway and should be just
fine. This doesn't really change the picture regarding glibc problem
when to many __thread are needed, but slightly improve the global
picture. Also this patch doesn't affect our performance in expedite
benchmark as far as I can tell.
This commit is contained in:
Cedric BAIL 2016-11-16 16:05:56 -08:00
parent 0b6e04d94f
commit 53dd596f43
7 changed files with 41 additions and 100 deletions

View File

@ -122,6 +122,10 @@ evas_common_draw_context_dup(RGBA_Draw_Context *dc)
pixman_image_ref(dc2->col.pixman_color_image);
#endif
#endif
dc2->cache.rects = NULL;
dc2->cache.used = 0;
return dc2;
}
@ -141,6 +145,7 @@ evas_common_draw_context_free(RGBA_Draw_Context *dc)
#endif
evas_common_draw_context_apply_clean_cutouts(&dc->cutout);
evas_common_draw_context_cutouts_real_free(dc->cache.rects);
free(dc);
}

View File

@ -348,12 +348,6 @@ error:
EAPI Eina_Bool
evas_common_font_draw_cb(RGBA_Image *dst, RGBA_Draw_Context *dc, int x, int y, Evas_Glyph_Array *glyphs, Evas_Common_Font_Draw_Cb cb)
{
#ifdef HAVE_THREAD_SPECIFIER
static __thread int rects_used = 0;
static __thread Cutout_Rects *rects = NULL;
#else
Cutout_Rects *rects = NULL;
#endif
int ext_x, ext_y, ext_w, ext_h;
int im_w, im_h;
RGBA_Gfx_Func func;
@ -408,26 +402,16 @@ evas_common_font_draw_cb(RGBA_Image *dst, RGBA_Draw_Context *dc, int x, int y, E
/* our clip is 0 size.. abort */
if ((dc->clip.w > 0) && (dc->clip.h > 0))
{
rects = evas_common_draw_context_apply_cutouts(dc, rects);
for (i = 0; i < rects->active; ++i)
dc->cache.rects = evas_common_draw_context_apply_cutouts(dc, dc->cache.rects);
for (i = 0; i < dc->cache.rects->active; ++i)
{
r = rects->rects + i;
r = dc->cache.rects->rects + i;
evas_common_draw_context_set_clip(dc, r->x, r->y, r->w, r->h);
ret |= cb(dst, dc, x, y, glyphs,
func, r->x, r->y, r->w, r->h,
im_w, im_h);
}
#ifdef HAVE_THREAD_SPECIFIER
rects_used++;
if (rects_used >= 4096)
{
evas_common_draw_context_cutouts_real_free(rects);
rects = NULL;
rects_used = 0;
}
#else
evas_common_draw_context_cutouts_real_free(rects);
#endif
evas_common_draw_context_cache_update(dc);
}
dc->clip.use = c; dc->clip.x = cx; dc->clip.y = cy; dc->clip.w = cw; dc->clip.h = ch;

View File

@ -752,12 +752,6 @@ evas_common_map_rgba_cb(RGBA_Image *src, RGBA_Image *dst,
int smooth, int level,
Evas_Common_Map_RGBA_Cb cb)
{
#ifdef HAVE_THREAD_SPECIFIER
static __thread int rects_used = 0;
static __thread Cutout_Rects *rects = NULL;
#else
Cutout_Rects *rects = NULL;
#endif
Cutout_Rect *r;
int c, cx, cy, cw, ch;
int i;
@ -789,24 +783,14 @@ evas_common_map_rgba_cb(RGBA_Image *src, RGBA_Image *dst,
dc->clip.use = c; dc->clip.x = cx; dc->clip.y = cy; dc->clip.w = cw; dc->clip.h = ch;
return;
}
rects = evas_common_draw_context_apply_cutouts(dc, rects);
for (i = 0; i < rects->active; ++i)
dc->cache.rects = evas_common_draw_context_apply_cutouts(dc, dc->cache.rects);
for (i = 0; i < dc->cache.rects->active; ++i)
{
r = rects->rects + i;
r = dc->cache.rects->rects + i;
evas_common_draw_context_set_clip(dc, r->x, r->y, r->w, r->h);
cb(src, dst, dc, p, smooth, level);
}
#ifdef HAVE_THREAD_SPECIFIER
rects_used++;
if (rects_used >= 4096)
{
evas_common_draw_context_cutouts_real_free(rects);
rects = NULL;
rects_used = 0;
}
#else
evas_common_draw_context_cutouts_real_free(rects);
#endif
evas_common_draw_context_cache_update(dc);
/* restore clip info */
dc->clip.use = c; dc->clip.x = cx; dc->clip.y = cy; dc->clip.w = cw; dc->clip.h = ch;
}
@ -814,12 +798,6 @@ evas_common_map_rgba_cb(RGBA_Image *src, RGBA_Image *dst,
EAPI Eina_Bool
evas_common_map_thread_rgba_cb(RGBA_Image *src, RGBA_Image *dst, RGBA_Draw_Context *dc, RGBA_Map *map, int smooth, int level, int offset, Evas_Common_Map_Thread_RGBA_Cb cb)
{
#ifdef HAVE_THREAD_SPECIFIER
static __thread int rects_used = 0;
static __thread Cutout_Rects *rects = NULL;
#else
Cutout_Rects *rects = NULL;
#endif
Cutout_Rect *r;
int c, cx, cy, cw, ch;
int i;
@ -854,24 +832,14 @@ evas_common_map_thread_rgba_cb(RGBA_Image *src, RGBA_Image *dst, RGBA_Draw_Conte
return EINA_FALSE;
}
rects = evas_common_draw_context_apply_cutouts(dc, rects);
for (i = 0; i < rects->active; ++i)
dc->cache.rects = evas_common_draw_context_apply_cutouts(dc, dc->cache.rects);
for (i = 0; i < dc->cache.rects->active; ++i)
{
r = rects->rects + i;
r = dc->cache.rects->rects + i;
evas_common_draw_context_set_clip(dc, r->x, r->y, r->w, r->h);
ret |= cb(src, dst, dc, map, smooth, level, offset);
}
#ifdef HAVE_THREAD_SPECIFIER
rects_used++;
if (rects_used >= 4096)
{
evas_common_draw_context_cutouts_real_free(rects);
rects = NULL;
rects_used = 0;
}
#else
evas_common_draw_context_cutouts_real_free(rects);
#endif
evas_common_draw_context_cache_update(dc);
/* restore clip info */
dc->clip.use = c; dc->clip.x = cx; dc->clip.y = cy; dc->clip.w = cw; dc->clip.h = ch;

View File

@ -12,12 +12,6 @@ evas_common_rectangle_init(void)
EAPI void
evas_common_rectangle_draw_cb(RGBA_Image *dst, RGBA_Draw_Context *dc, int x, int y, int w, int h, Evas_Common_Rectangle_Draw_Cb cb)
{
#ifdef HAVE_THREAD_SPECIFIER
static __thread int rects_used = 0;
static __thread Cutout_Rects *rects = NULL;
#else
Cutout_Rects *rects = NULL;
#endif
Cutout_Rect *r;
int c, cx, cy, cw, ch;
int i;
@ -40,24 +34,14 @@ evas_common_rectangle_draw_cb(RGBA_Image *dst, RGBA_Draw_Context *dc, int x, int
/* our clip is 0 size.. abort */
if ((dc->clip.w > 0) && (dc->clip.h > 0))
{
rects = evas_common_draw_context_apply_cutouts(dc, rects);
for (i = 0; i < rects->active; ++i)
dc->cache.rects = evas_common_draw_context_apply_cutouts(dc, dc->cache.rects);
for (i = 0; i < dc->cache.rects->active; ++i)
{
r = rects->rects + i;
r = dc->cache.rects->rects + i;
evas_common_draw_context_set_clip(dc, r->x, r->y, r->w, r->h);
cb(dst, dc, x, y, w, h);
}
#ifdef HAVE_THREAD_SPECIFIER
rects_used++;
if (rects_used >= 4096)
{
evas_common_draw_context_cutouts_real_free(rects);
rects = NULL;
rects_used = 0;
}
#else
evas_common_draw_context_cutouts_real_free(rects);
#endif
evas_common_draw_context_cache_update(dc);
}
}
/* restore clip info */

View File

@ -40,12 +40,6 @@ evas_common_scale_rgba_in_to_out_clip_cb(RGBA_Image *src, RGBA_Image *dst,
int dst_region_w, int dst_region_h,
Evas_Common_Scale_In_To_Out_Clip_Cb cb)
{
#ifdef HAVE_THREAD_SPECIFIER
static __thread int rects_used = 0;
static __thread Cutout_Rects *rects = NULL;
#else
Cutout_Rects *rects = NULL;
#endif
Cutout_Rect *r;
int c, cx, cy, cw, ch;
int i;
@ -76,26 +70,16 @@ evas_common_scale_rgba_in_to_out_clip_cb(RGBA_Image *src, RGBA_Image *dst,
return EINA_FALSE;
}
rects = evas_common_draw_context_apply_cutouts(dc, rects);
for (i = 0; i < rects->active; ++i)
dc->cache.rects = evas_common_draw_context_apply_cutouts(dc, dc->cache.rects);
for (i = 0; i < dc->cache.rects->active; ++i)
{
r = rects->rects + i;
r = dc->cache.rects->rects + i;
evas_common_draw_context_set_clip(dc, r->x, r->y, r->w, r->h);
ret |= cb(src, dst, dc,
src_region_x, src_region_y, src_region_w, src_region_h,
dst_region_x, dst_region_y, dst_region_w, dst_region_h);
}
#ifdef HAVE_THREAD_SPECIFIER
rects_used++;
if (rects_used >= 4096)
{
evas_common_draw_context_cutouts_real_free(rects);
rects = NULL;
rects_used = 0;
}
#else
evas_common_draw_context_cutouts_real_free(rects);
#endif
evas_common_draw_context_cache_update(dc);
/* restore clip info */
dc->clip.use = c; dc->clip.x = cx; dc->clip.y = cy; dc->clip.w = cw; dc->clip.h = ch;

View File

@ -756,6 +756,10 @@ struct _RGBA_Draw_Context
Eina_Bool async : 1;
} clip;
Cutout_Rects cutout;
struct {
Cutout_Rects *rects;
int used;
} cache;
struct {
struct {
void *(*gl_new) (void *data, RGBA_Font_Glyph *fg);

View File

@ -427,6 +427,18 @@ evas_canvas_async_block(Evas_Public_Data *e)
}
}
static inline void
evas_common_draw_context_cache_update(RGBA_Draw_Context *dc)
{
dc->cache.used++;
if (dc->cache.used >= 4096)
{
evas_common_draw_context_cutouts_real_free(dc->cache.rects);
dc->cache.rects = NULL;
dc->cache.used = 0;
}
}
#define _EVAS_COLOR_CLAMP(x, y) do { \
if (x > y) { x = y; bad = 1; } \
if (x < 0) { x = 0; bad = 1; } } while (0)