#include "evas_common.h" #include "evas_private.h" /* ***** ** ** ENGINE ROUTINES ** ***** */ static int cpunum = 0; static void * eng_context_new(void *data) { return evas_common_draw_context_new(); } static void eng_context_free(void *data, void *context) { evas_common_draw_context_free(context); } static void eng_context_clip_set(void *data, void *context, int x, int y, int w, int h) { evas_common_draw_context_set_clip(context, x, y, w, h); } static void eng_context_clip_clip(void *data, void *context, int x, int y, int w, int h) { evas_common_draw_context_clip_clip(context, x, y, w, h); } static void eng_context_clip_unset(void *data, void *context) { evas_common_draw_context_unset_clip(context); } static int eng_context_clip_get(void *data, void *context, int *x, int *y, int *w, int *h) { *x = ((RGBA_Draw_Context *)context)->clip.x; *y = ((RGBA_Draw_Context *)context)->clip.y; *w = ((RGBA_Draw_Context *)context)->clip.w; *h = ((RGBA_Draw_Context *)context)->clip.h; return ((RGBA_Draw_Context *)context)->clip.use; } static void eng_context_color_set(void *data, void *context, int r, int g, int b, int a) { evas_common_draw_context_set_color(context, r, g, b, a); } static int eng_context_color_get(void *data, void *context, int *r, int *g, int *b, int *a) { *r = (int)(R_VAL(&((RGBA_Draw_Context *)context)->col.col)); *g = (int)(G_VAL(&((RGBA_Draw_Context *)context)->col.col)); *b = (int)(B_VAL(&((RGBA_Draw_Context *)context)->col.col)); *a = (int)(A_VAL(&((RGBA_Draw_Context *)context)->col.col)); return 1; } static void eng_context_multiplier_set(void *data, void *context, int r, int g, int b, int a) { evas_common_draw_context_set_multiplier(context, r, g, b, a); } static void eng_context_multiplier_unset(void *data, void *context) { evas_common_draw_context_unset_multiplier(context); } static int eng_context_multiplier_get(void *data, void *context, int *r, int *g, int *b, int *a) { *r = (int)(R_VAL(&((RGBA_Draw_Context *)context)->mul.col)); *g = (int)(G_VAL(&((RGBA_Draw_Context *)context)->mul.col)); *b = (int)(B_VAL(&((RGBA_Draw_Context *)context)->mul.col)); *a = (int)(A_VAL(&((RGBA_Draw_Context *)context)->mul.col)); return ((RGBA_Draw_Context *)context)->mul.use; } static void eng_context_cutout_add(void *data, void *context, int x, int y, int w, int h) { evas_common_draw_context_add_cutout(context, x, y, w, h); } static void eng_context_cutout_clear(void *data, void *context) { evas_common_draw_context_clear_cutouts(context); } static void eng_context_anti_alias_set(void *data, void *context, unsigned char aa) { evas_common_draw_context_set_anti_alias(context, aa); } static unsigned char eng_context_anti_alias_get(void *data, void *context) { return ((RGBA_Draw_Context *)context)->anti_alias; } static void eng_context_color_interpolation_set(void *data, void *context, int color_space) { evas_common_draw_context_set_color_interpolation(context, color_space); } static int eng_context_color_interpolation_get(void *data, void *context) { return ((RGBA_Draw_Context *)context)->interpolation.color_space; } static void eng_context_render_op_set(void *data, void *context, int op) { evas_common_draw_context_set_render_op(context, op); } static int eng_context_render_op_get(void *data, void *context) { return ((RGBA_Draw_Context *)context)->render_op; } static void eng_rectangle_draw(void *data, void *context, void *surface, int x, int y, int w, int h) { #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_rectangle_draw(surface, context, x, y, w, h); else #endif { evas_common_rectangle_draw(surface, context, x, y, w, h); evas_common_cpu_end_opt(); } } static void eng_line_draw(void *data, void *context, void *surface, int x1, int y1, int x2, int y2) { #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_line_draw(surface, context, x1, y1, x2, y2); else #endif { evas_common_line_draw(surface, context, x1, y1, x2, y2); evas_common_cpu_end_opt(); } } static void * eng_polygon_point_add(void *data, void *context, void *polygon, int x, int y) { return evas_common_polygon_point_add(polygon, x, y); } static void * eng_polygon_points_clear(void *data, void *context, void *polygon) { return evas_common_polygon_points_clear(polygon); } static void eng_polygon_draw(void *data, void *context, void *surface, void *polygon) { #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_poly_draw(surface, context, polygon); else #endif { evas_common_polygon_draw(surface, context, polygon); evas_common_cpu_end_opt(); } } static void * eng_gradient_new(void *data) { return evas_common_gradient_new(); } static void eng_gradient_free(void *data, void *gradient) { evas_common_gradient_free(gradient); } static void eng_gradient_color_stop_add(void *data, void *gradient, int r, int g, int b, int a, int delta) { evas_common_gradient_color_stop_add(gradient, r, g, b, a, delta); } static void eng_gradient_alpha_stop_add(void *data, void *gradient, int a, int delta) { evas_common_gradient_alpha_stop_add(gradient, a, delta); } static void eng_gradient_color_data_set(void *data, void *gradient, void *map, int len, int has_alpha) { evas_common_gradient_color_data_set(gradient, map, len, has_alpha); } static void eng_gradient_alpha_data_set(void *data, void *gradient, void *alpha_map, int len) { evas_common_gradient_alpha_data_set(gradient, alpha_map, len); } static void eng_gradient_clear(void *data, void *gradient) { evas_common_gradient_clear(gradient); } static void eng_gradient_fill_set(void *data, void *gradient, int x, int y, int w, int h) { evas_common_gradient_fill_set(gradient, x, y, w, h); } static void eng_gradient_fill_angle_set(void *data, void *gradient, double angle) { evas_common_gradient_fill_angle_set(gradient, angle); } static void eng_gradient_fill_spread_set(void *data, void *gradient, int spread) { evas_common_gradient_fill_spread_set(gradient, spread); } static void eng_gradient_angle_set(void *data, void *gradient, double angle) { evas_common_gradient_map_angle_set(gradient, angle); } static void eng_gradient_offset_set(void *data, void *gradient, float offset) { evas_common_gradient_map_offset_set(gradient, offset); } static void eng_gradient_direction_set(void *data, void *gradient, int direction) { evas_common_gradient_map_direction_set(gradient, direction); } static void eng_gradient_type_set(void *data, void *gradient, char *name, char *params) { evas_common_gradient_type_set(gradient, name, params); } static int eng_gradient_is_opaque(void *data, void *context, void *gradient, int x, int y, int w, int h) { RGBA_Draw_Context *dc = (RGBA_Draw_Context *)context; RGBA_Gradient *gr = (RGBA_Gradient *)gradient; if (!dc || !gr || !gr->type.geometer) return 0; return !(gr->type.geometer->has_alpha(gr, dc->render_op) | gr->type.geometer->has_mask(gr, dc->render_op)); } static int eng_gradient_is_visible(void *data, void *context, void *gradient, int x, int y, int w, int h) { RGBA_Draw_Context *dc = (RGBA_Draw_Context *)context; if (!dc || !gradient) return 0; return 1; } static void eng_gradient_render_pre(void *data, void *context, void *gradient) { RGBA_Draw_Context *dc = (RGBA_Draw_Context *)context; RGBA_Gradient *gr = (RGBA_Gradient *)gradient; int len; if (!dc || !gr || !gr->type.geometer) return; gr->type.geometer->geom_set(gr); len = gr->type.geometer->get_map_len(gr); evas_common_gradient_map(dc, gr, len); } static void eng_gradient_render_post(void *data, void *gradient) { } static void eng_gradient_draw(void *data, void *context, void *surface, void *gradient, int x, int y, int w, int h) { #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_grad_draw(surface, context, x, y, w, h, gradient); else #endif { evas_common_gradient_draw(surface, context, x, y, w, h, gradient); evas_common_cpu_end_opt(); } } static int eng_image_alpha_get(void *data, void *image) { Image_Entry *im; if (!image) return 1; im = image; switch (im->space) { case EVAS_COLORSPACE_ARGB8888: if (im->flags.alpha) return 1; default: break; } return 0; } static int eng_image_colorspace_get(void *data, void *image) { Image_Entry *im; if (!image) return EVAS_COLORSPACE_ARGB8888; im = image; return im->space; } static void * eng_image_alpha_set(void *data, void *image, int has_alpha) { RGBA_Image *im; if (!image) return NULL; im = image; if (im->cache_entry.space != EVAS_COLORSPACE_ARGB8888) { im->cache_entry.flags.alpha = 0; return im; } im = (RGBA_Image *) evas_cache_image_alone(&im->cache_entry); evas_common_image_colorspace_dirty(im); im->cache_entry.flags.alpha = has_alpha ? 1 : 0; return im; } static void * eng_image_border_set(void *data, void *image, int l, int r, int t, int b) { RGBA_Image *im; im = image; return im; } static void eng_image_border_get(void *data, void *image, int *l, int *r, int *t, int *b) { RGBA_Image *im; im = image; } static char * eng_image_comment_get(void *data, void *image, char *key) { RGBA_Image *im; if (!image) return NULL; im = image; return im->info.comment; } static char * eng_image_format_get(void *data, void *image) { return NULL; } static void eng_image_colorspace_set(void *data, void *image, int cspace) { Image_Entry *im; if (!image) return; im = image; evas_cache_image_colorspace(im, cspace); } static void eng_image_native_set(void *data, void *image, void *native) { } static void * eng_image_native_get(void *data, void *image) { return NULL; } static void * eng_image_load(void *data, const char *file, const char *key, int *error, Evas_Image_Load_Opts *lo) { *error = 0; return evas_common_load_image_from_file(file, key, lo); } static void * eng_image_new_from_data(void *data, int w, int h, DATA32 *image_data, int alpha, int cspace) { return evas_cache_image_data(evas_common_image_cache_get(), w, h, image_data, alpha, cspace); } static void * eng_image_new_from_copied_data(void *data, int w, int h, DATA32 *image_data, int alpha, int cspace) { return evas_cache_image_copied_data(evas_common_image_cache_get(), w, h, image_data, alpha, cspace); } static void eng_image_free(void *data, void *image) { evas_cache_image_drop(image); } static void eng_image_size_get(void *data, void *image, int *w, int *h) { Image_Entry *im; im = image; if (w) *w = im->w; if (h) *h = im->h; } static void * eng_image_size_set(void *data, void *image, int w, int h) { Image_Entry *im; im = image; return evas_cache_image_size_set(image, w, h); } static void * eng_image_dirty_region(void *data, void *image, int x, int y, int w, int h) { Image_Entry *im = image; if (!image) return NULL; return evas_cache_image_dirty(im, x, y, w, h); } static void * eng_image_data_get(void *data, void *image, int to_write, DATA32 **image_data) { RGBA_Image *im; if (!image) { *image_data = NULL; return NULL; } im = image; evas_cache_image_load_data(&im->cache_entry); switch (im->cache_entry.space) { case EVAS_COLORSPACE_ARGB8888: if (to_write) im = (RGBA_Image *) evas_cache_image_alone(&im->cache_entry); *image_data = im->image.data; break; case EVAS_COLORSPACE_YCBCR422P601_PL: case EVAS_COLORSPACE_YCBCR422P709_PL: *image_data = im->cs.data; break; default: abort(); break; } return im; } static void * eng_image_data_put(void *data, void *image, DATA32 *image_data) { RGBA_Image *im, *im2; if (!image) return NULL; im = image; switch (im->cache_entry.space) { case EVAS_COLORSPACE_ARGB8888: if (image_data != im->image.data) { int w, h; w = im->cache_entry.w; h = im->cache_entry.h; im2 = eng_image_new_from_data(data, w, h, image_data, eng_image_alpha_get(data, image), eng_image_colorspace_get(data, image)); evas_cache_image_drop(&im->cache_entry); im = im2; } break; case EVAS_COLORSPACE_YCBCR422P601_PL: case EVAS_COLORSPACE_YCBCR422P709_PL: if (image_data != im->cs.data) { if (im->cs.data) { if (!im->cs.no_free) free(im->cs.data); } im->cs.data = image_data; evas_common_image_colorspace_dirty(im); } break; default: abort(); break; } return im; } static void eng_image_draw(void *data, void *context, void *surface, void *image, int src_x, int src_y, int src_w, int src_h, int dst_x, int dst_y, int dst_w, int dst_h, int smooth) { RGBA_Image *im; if (!image) return; im = image; if (im->cache_entry.space == EVAS_COLORSPACE_ARGB8888) evas_cache_image_load_data(&im->cache_entry); evas_common_image_colorspace_normalize(im); #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_image_draw(im, surface, context, smooth, src_x, src_y, src_w, src_h, dst_x, dst_y, dst_w, dst_h); else #endif { if (smooth) evas_common_scale_rgba_in_to_out_clip_smooth(im, surface, context, src_x, src_y, src_w, src_h, dst_x, dst_y, dst_w, dst_h); else evas_common_scale_rgba_in_to_out_clip_sample(im, surface, context, src_x, src_y, src_w, src_h, dst_x, dst_y, dst_w, dst_h); evas_common_cpu_end_opt(); } } static void eng_image_cache_flush(void *data) { int tmp_size; tmp_size = evas_common_image_get_cache(); evas_common_image_set_cache(0); evas_common_image_set_cache(tmp_size); } static void eng_image_cache_set(void *data, int bytes) { evas_common_image_set_cache(bytes); } static int eng_image_cache_get(void *data) { return evas_common_image_get_cache(); } static void * eng_font_load(void *data, const char *name, int size) { return evas_common_font_load(name, size); } static void * eng_font_memory_load(void *data, char *name, int size, const void *fdata, int fdata_size) { return evas_common_font_memory_load(name, size, fdata, fdata_size); } static void * eng_font_add(void *data, void *font, const char *name, int size) { return evas_common_font_add(font, name, size); } static void * eng_font_memory_add(void *data, void *font, char *name, int size, const void *fdata, int fdata_size) { return evas_common_font_memory_add(font, name, size, fdata, fdata_size); } static void eng_font_free(void *data, void *font) { evas_common_font_free(font); } static int eng_font_ascent_get(void *data, void *font) { return evas_common_font_ascent_get(font); } static int eng_font_descent_get(void *data, void *font) { return evas_common_font_descent_get(font); } static int eng_font_max_ascent_get(void *data, void *font) { return evas_common_font_max_ascent_get(font); } static int eng_font_max_descent_get(void *data, void *font) { return evas_common_font_max_descent_get(font); } static void eng_font_string_size_get(void *data, void *font, const char *text, int *w, int *h) { evas_common_font_query_size(font, text, w, h); } static int eng_font_inset_get(void *data, void *font, const char *text) { return evas_common_font_query_inset(font, text); } static int eng_font_h_advance_get(void *data, void *font, const char *text) { int h, v; evas_common_font_query_advance(font, text, &h, &v); return h; } static int eng_font_v_advance_get(void *data, void *font, const char *text) { int h, v; evas_common_font_query_advance(font, text, &h, &v); return v; } static int eng_font_char_coords_get(void *data, void *font, const char *text, int pos, int *cx, int *cy, int *cw, int *ch) { return evas_common_font_query_char_coords(font, text, pos, cx, cy, cw, ch); } static int eng_font_char_at_coords_get(void *data, void *font, const char *text, int x, int y, int *cx, int *cy, int *cw, int *ch) { return evas_common_font_query_text_at_pos(font, text, x, y, cx, cy, cw, ch); } static void eng_font_draw(void *data, void *context, void *surface, void *font, int x, int y, int w, int h, int ow, int oh, const char *text) { #ifdef BUILD_PTHREAD if (cpunum > 1) evas_common_pipe_text_draw(surface, context, font, x, y, text); else #endif { evas_common_font_draw(surface, context, font, x, y, text); evas_common_cpu_end_opt(); } } static void eng_font_cache_flush(void *data) { evas_common_font_flush(); } static void eng_font_cache_set(void *data, int bytes) { evas_common_font_cache_set(bytes); } static int eng_font_cache_get(void *data) { return evas_common_font_cache_get(); } static void eng_font_hinting_set(void *data, void *font, int hinting) { evas_common_font_hinting_set(font, hinting); } static int eng_font_hinting_can_hint(void *data, int hinting) { return evas_common_hinting_available(hinting); } /* ***** ** ** ENGINE API ** ***** */ static Evas_Func func = { NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, NULL, /* draw context virtual methods */ eng_context_new, eng_context_free, eng_context_clip_set, eng_context_clip_clip, eng_context_clip_unset, eng_context_clip_get, eng_context_color_set, eng_context_color_get, eng_context_multiplier_set, eng_context_multiplier_unset, eng_context_multiplier_get, eng_context_cutout_add, eng_context_cutout_clear, eng_context_anti_alias_set, eng_context_anti_alias_get, eng_context_color_interpolation_set, eng_context_color_interpolation_get, eng_context_render_op_set, eng_context_render_op_get, /* rect draw funcs */ eng_rectangle_draw, /* line draw funcs */ eng_line_draw, /* polygon draw funcs */ eng_polygon_point_add, eng_polygon_points_clear, eng_polygon_draw, /* gradient draw funcs */ eng_gradient_new, eng_gradient_free, eng_gradient_color_stop_add, eng_gradient_alpha_stop_add, eng_gradient_color_data_set, eng_gradient_alpha_data_set, eng_gradient_clear, eng_gradient_fill_set, eng_gradient_fill_angle_set, eng_gradient_fill_spread_set, eng_gradient_angle_set, eng_gradient_offset_set, eng_gradient_direction_set, eng_gradient_type_set, eng_gradient_is_opaque, eng_gradient_is_visible, eng_gradient_render_pre, eng_gradient_render_post, eng_gradient_draw, /* image draw funcs */ eng_image_load, eng_image_new_from_data, eng_image_new_from_copied_data, eng_image_free, eng_image_size_get, eng_image_size_set, NULL, eng_image_dirty_region, eng_image_data_get, eng_image_data_put, eng_image_alpha_set, eng_image_alpha_get, eng_image_border_set, eng_image_border_get, eng_image_draw, eng_image_comment_get, eng_image_format_get, eng_image_colorspace_set, eng_image_colorspace_get, eng_image_native_set, eng_image_native_get, /* image cache funcs */ eng_image_cache_flush, eng_image_cache_set, eng_image_cache_get, /* font draw functions */ eng_font_load, eng_font_memory_load, eng_font_add, eng_font_memory_add, eng_font_free, eng_font_ascent_get, eng_font_descent_get, eng_font_max_ascent_get, eng_font_max_descent_get, eng_font_string_size_get, eng_font_inset_get, eng_font_h_advance_get, eng_font_v_advance_get, eng_font_char_coords_get, eng_font_char_at_coords_get, eng_font_draw, /* font cache functions */ eng_font_cache_flush, eng_font_cache_set, eng_font_cache_get, /* font hinting functions */ eng_font_hinting_set, eng_font_hinting_can_hint /* FUTURE software generic calls go here */ }; /* ***** ** ** MODULE ACCESSIBLE API API ** ***** */ EAPI int module_open(Evas_Module *em) { if (!em) return 0; em->functions = (void *)(&func); cpunum = evas_common_cpu_count(); return 1; } EAPI void module_close(void) { } EAPI Evas_Module_Api evas_modapi = { EVAS_MODULE_API_VERSION, EVAS_MODULE_TYPE_ENGINE, "software_generic", "none" };