This source file includes following definitions.
- get_surface_world_matrix
- evg_surface_new
- evg_surface_delete
- evg_surface_set_matrix
- evg_surface_attach_to_callbacks
- evg_surface_attach_to_buffer
- evg_surface_attach_to_texture
- evg_surface_detach
- evg_surface_clear
- evg_surface_set_raster_level
- evg_surface_set_clipper
- setup_grey_callback
- evg_surface_set_path
- evg_surface_fill
#include "rast_soft.h"
static void get_surface_world_matrix(EVGSurface *_this, GF_Matrix2D *mat)
{
gf_mx2d_init(*mat);
if (_this->center_coords) {
gf_mx2d_add_scale(mat, FIX_ONE, -FIX_ONE);
gf_mx2d_add_translation(mat, INT2FIX(_this->width / 2), INT2FIX(_this->height / 2));
}
}
GF_SURFACE evg_surface_new(GF_Raster2D *_dr, Bool center_coords)
{
EVGSurface *_this;
GF_SAFEALLOC(_this, EVGSurface);
if (_this) {
_this->center_coords = center_coords;
_this->texture_filter = GF_TEXTURE_FILTER_DEFAULT;
_this->ftparams.source = &_this->ftoutline;
_this->ftparams.user = _this;
_this->raster = evg_raster_new();
}
return _this;
}
void evg_surface_delete(GF_SURFACE _this)
{
EVGSurface *surf = (EVGSurface *)_this;
if (!surf)
return;
#ifndef INLINE_POINT_CONVERSION
if (surf->points) gf_free(surf->points);
surf->points = NULL;
#endif
if (surf->stencil_pix_run) gf_free(surf->stencil_pix_run);
surf->stencil_pix_run = NULL;
if (surf->raster) evg_raster_del(surf->raster);
surf->raster = NULL;
gf_free(surf);
}
GF_Err evg_surface_set_matrix(GF_SURFACE _this, GF_Matrix2D *mat)
{
GF_Matrix2D tmp;
EVGSurface *surf = (EVGSurface *)_this;
if (!surf) return GF_BAD_PARAM;
get_surface_world_matrix(surf, &surf->mat);
if (!mat) return GF_OK;
gf_mx2d_init(tmp);
gf_mx2d_add_matrix(&tmp, mat);
gf_mx2d_add_matrix(&tmp, &surf->mat);
gf_mx2d_copy(surf->mat, tmp);
return GF_OK;
}
GF_Err evg_surface_attach_to_callbacks(GF_SURFACE _this, GF_RasterCallback *callbacks, u32 width, u32 height)
{
EVGSurface *surf = (EVGSurface *)_this;
if (!surf || !width || !height || !callbacks) return GF_BAD_PARAM;
if (!callbacks->cbk || !callbacks->fill_run_alpha || !callbacks->fill_run_no_alpha || !callbacks->fill_rect) return GF_BAD_PARAM;
surf->width = width;
surf->height = height;
if (surf->stencil_pix_run) gf_free(surf->stencil_pix_run);
surf->stencil_pix_run = (u32 *) gf_malloc(sizeof(u32) * (width+2));
surf->raster_cbk = callbacks->cbk;
surf->raster_fill_run_alpha = callbacks->fill_run_alpha;
surf->raster_fill_run_no_alpha = callbacks->fill_run_no_alpha;
surf->raster_fill_rectangle = callbacks->fill_rect;
evg_surface_set_matrix(surf, NULL);
return GF_OK;
}
GF_Err evg_surface_attach_to_buffer(GF_SURFACE _this, char *pixels, u32 width, u32 height, s32 pitch_x, s32 pitch_y, GF_PixelFormat pixelFormat)
{
u32 BPP;
EVGSurface *surf = (EVGSurface *)_this;
if (!surf || !pixels || (pixelFormat>GF_PIXEL_YUVA)) return GF_BAD_PARAM;
switch (pixelFormat) {
#ifdef GF_RGB_444_SUPORT
case GF_PIXEL_RGB_444:
#endif
#ifdef GF_RGB_555_SUPORT
case GF_PIXEL_RGB_555:
#endif
case GF_PIXEL_RGB_565:
BPP = 2;
break;
case GF_PIXEL_BGR_24:
case GF_PIXEL_RGB_24:
BPP = 3;
break;
case GF_PIXEL_RGB_32:
case GF_PIXEL_ARGB:
case GF_PIXEL_RGBA:
case GF_PIXEL_BGR_32:
BPP = 4;
break;
default:
return GF_NOT_SUPPORTED;
}
if (!pitch_x) pitch_x = BPP;
surf->pitch_x = pitch_x;
surf->pitch_y = pitch_y;
if (!surf->stencil_pix_run || (surf->width != width)) {
if (surf->stencil_pix_run) gf_free(surf->stencil_pix_run);
surf->stencil_pix_run = (u32 *) gf_malloc(sizeof(u32) * (width+2));
}
surf->width = width;
surf->height = height;
surf->pixels = (char*)pixels;
surf->pixelFormat = pixelFormat;
surf->BPP = BPP;
surf->raster_cbk = NULL;
surf->raster_fill_run_alpha = NULL;
surf->raster_fill_run_no_alpha = NULL;
evg_surface_set_matrix(_this, NULL);
return GF_OK;
}
GF_Err evg_surface_attach_to_texture(GF_SURFACE _this, GF_STENCIL sten)
{
u32 BPP;
EVGSurface *surf = (EVGSurface *)_this;
EVG_Texture *tx = (EVG_Texture *) sten;;
if (!surf || (tx->type != GF_STENCIL_TEXTURE)) return GF_BAD_PARAM;
switch (tx->pixel_format) {
case GF_PIXEL_GREYSCALE:
BPP = 1;
break;
case GF_PIXEL_ALPHAGREY:
#ifdef GF_RGB_444_SUPORT
case GF_PIXEL_RGB_444:
#endif
#ifdef GF_RGB_555_SUPORT
case GF_PIXEL_RGB_555:
#endif
case GF_PIXEL_RGB_565:
BPP = 2;
break;
case GF_PIXEL_BGR_24:
case GF_PIXEL_RGB_24:
BPP = 3;
break;
case GF_PIXEL_RGB_32:
case GF_PIXEL_BGR_32:
case GF_PIXEL_ARGB:
case GF_PIXEL_RGBA:
BPP = 4;
break;
default:
return GF_NOT_SUPPORTED;
}
surf->pitch_x = BPP;
surf->pitch_y = tx->stride;
if (surf->stencil_pix_run) gf_free(surf->stencil_pix_run);
surf->stencil_pix_run = (u32 *) gf_malloc(sizeof(u32) * (tx->width+2));
surf->width = tx->width;
surf->height = tx->height;
surf->pixels = tx->pixels;
surf->pixelFormat = tx->pixel_format;
surf->BPP = BPP;
surf->raster_cbk = NULL;
surf->raster_fill_run_alpha = NULL;
surf->raster_fill_run_no_alpha = NULL;
evg_surface_set_matrix(surf, NULL);
return GF_OK;
}
void evg_surface_detach(GF_SURFACE _this)
{
EVGSurface *surf = (EVGSurface *)_this;
surf->raster_cbk = NULL;
surf->raster_fill_run_alpha = NULL;
surf->raster_fill_run_no_alpha = NULL;
}
GF_Err evg_surface_clear(GF_SURFACE _this, GF_IRect *rc, u32 color)
{
GF_IRect clear;
EVGSurface *surf = (EVGSurface *)_this;
if (!surf) return GF_BAD_PARAM;
if (rc) {
s32 _x, _y;
if (surf->center_coords) {
_x = rc->x + surf->width / 2;
_y = surf->height / 2 - rc->y;
} else {
_x = rc->x;
_y = rc->y - rc->height;
}
clear.width = (u32) rc->width;
if (_x>=0) {
clear.x = (u32) _x;
} else {
if ( (s32) clear.width + _x < 0) return GF_BAD_PARAM;
clear.width += _x;
clear.x = 0;
}
clear.height = (u32) rc->height;
if (_y>=0) {
clear.y = _y;
} else {
if ( (s32) clear.height + _y < 0) return GF_BAD_PARAM;
clear.height += _y;
clear.y = 0;
}
} else {
clear.x = clear.y = 0;
clear.width = surf->width;
clear.height = surf->height;
}
if (surf->raster_cbk) {
surf->raster_fill_rectangle(surf->raster_cbk, clear.x, clear.y, clear.width, clear.height, color);
return GF_OK;
}
switch (surf->pixelFormat) {
case GF_PIXEL_ARGB:
case GF_PIXEL_RGB_32:
return evg_surface_clear_bgra(surf, clear, color);
case GF_PIXEL_BGR_32:
return evg_surface_clear_rgbx(surf, clear, color);
case GF_PIXEL_RGBA:
return evg_surface_clear_rgba(surf, clear, color);
case GF_PIXEL_BGR_24:
return evg_surface_clear_bgr(surf, clear, color);
case GF_PIXEL_RGB_24:
return evg_surface_clear_rgb(surf, clear, color);
case GF_PIXEL_RGB_565:
return evg_surface_clear_565(surf, clear, color);
#ifdef GF_RGB_444_SUPORT
case GF_PIXEL_RGB_444:
return evg_surface_clear_444(surf, clear, color);
#endif
#ifdef GF_RGB_555_SUPORT
case GF_PIXEL_RGB_555:
return evg_surface_clear_555(surf, clear, color);
#endif
default:
return GF_BAD_PARAM;
}
}
GF_Err evg_surface_set_raster_level(GF_SURFACE _this, GF_RasterLevel RasterSetting)
{
EVGSurface *surf = (EVGSurface *)_this;
if (!surf) return GF_BAD_PARAM;
switch (RasterSetting) {
case GF_RASTER_HIGH_QUALITY:
surf->texture_filter = GF_TEXTURE_FILTER_HIGH_QUALITY;
break;
case GF_RASTER_MID:
surf->texture_filter = GF_TEXTURE_FILTER_HIGH_QUALITY;
break;
case GF_RASTER_HIGH_SPEED:
default:
surf->texture_filter = GF_TEXTURE_FILTER_HIGH_SPEED;
break;
}
return GF_OK;
}
GF_Err evg_surface_set_clipper(GF_SURFACE _this , GF_IRect *rc)
{
EVGSurface *surf = (EVGSurface *)_this;
if (!surf) return GF_BAD_PARAM;
if (rc) {
surf->clipper = *rc;
surf->useClipper = 1;
if (surf->center_coords) {
surf->clipper.x += surf->width / 2;
surf->clipper.y = surf->height / 2 - rc->y;
} else {
surf->clipper.y -= rc->height;
}
if (surf->clipper.x <=0) {
if (surf->clipper.x + (s32) surf->clipper.width < 0) return GF_BAD_PARAM;
surf->clipper.width += surf->clipper.x;
surf->clipper.x = 0;
}
if (surf->clipper.y <=0) {
if (surf->clipper.y + (s32) surf->clipper.height < 0) return GF_BAD_PARAM;
surf->clipper.height += surf->clipper.y;
surf->clipper.y = 0;
}
if (surf->clipper.x + surf->clipper.width > (s32) surf->width) {
surf->clipper.width = surf->width - surf->clipper.x;
}
if (surf->clipper.y + surf->clipper.height > (s32) surf->height) {
surf->clipper.height = surf->height - surf->clipper.y;
}
} else {
surf->useClipper = 0;
}
return GF_OK;
}
static Bool setup_grey_callback(EVGSurface *surf)
{
u32 col, a;
Bool use_const = GF_TRUE;
if (surf->sten->type == GF_STENCIL_SOLID) {
col = surf->fill_col = ((EVG_Brush *)surf->sten)->color;
a = GF_COL_A(surf->fill_col);
} else {
col = a = 0;
use_const = GF_FALSE;
}
if (surf->raster_cbk) {
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_user_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_user_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_user_fill_var;
}
return GF_TRUE;
}
switch (surf->pixelFormat) {
case GF_PIXEL_ARGB:
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgra_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgra_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgra_fill_var;
}
break;
case GF_PIXEL_RGBA:
if (use_const) {
if (!a) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgba_fill_erase;
} else if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgba_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgba_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgba_fill_var;
}
break;
case GF_PIXEL_RGB_32:
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgrx_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgrx_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgrx_fill_var;
}
break;
case GF_PIXEL_BGR_32:
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgbx_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgbx_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgbx_fill_var;
}
break;
case GF_PIXEL_RGB_24:
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgb_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgb_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_rgb_fill_var;
}
break;
case GF_PIXEL_BGR_24:
if (use_const) {
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgr_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgr_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_bgr_fill_var;
}
break;
case GF_PIXEL_RGB_565:
if (use_const) {
surf->fill_565 = GF_COL_TO_565(col);
if (!a) return GF_FALSE;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_565_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_565_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_565_fill_var;
assert(surf->sten->fill_run);
}
break;
#ifdef GF_RGB_444_SUPORT
if (use_const) {
surf->fill_444 = GF_COL_TO_444(col);
if (!a) return 0;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_444_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_444_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_444_fill_var;
}
break;
#endif
#ifdef GF_RGB_555_SUPORT
case GF_PIXEL_RGB_555:
if (use_const) {
surf->fill_555 = GF_COL_TO_555(col);
if (!a) return 0;
if (a!=0xFF) {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_555_fill_const_a;
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_555_fill_const;
}
} else {
surf->ftparams.gray_spans = (EVG_Raster_Span_Func) evg_555_fill_var;
}
break;
#endif
}
return GF_TRUE;
}
GF_Err evg_surface_set_path(GF_SURFACE _this, GF_Path *gp)
{
#ifndef INLINE_POINT_CONVERSION
u32 i;
GF_Point2D pt;
#endif
EVGSurface *surf = (EVGSurface *)_this;
if (!surf) return GF_BAD_PARAM;
if (!gp || !gp->n_points) {
surf->ftoutline.n_points = 0;
surf->ftoutline.n_contours = 0;
return GF_OK;
}
gf_path_flatten(gp);
surf->ftoutline.n_points = gp->n_points;
surf->ftoutline.n_contours = gp->n_contours;
surf->ftoutline.tags = gp->tags;
surf->ftoutline.contours = (s32*) gp->contours;
gf_path_get_bounds(gp, &surf->path_bounds);
surf->path_bounds.y -= surf->path_bounds.height;
surf->ftoutline.flags = 0;
if (gp->flags & GF_PATH_FILL_ZERO_NONZERO) surf->ftoutline.flags = GF_PATH_FILL_ZERO_NONZERO;
#ifdef INLINE_POINT_CONVERSION
surf->ftoutline.n_points = gp->n_points;
surf->ftoutline.points = gp->points;
surf->ftparams.mx = &surf->mat;
#else
if (surf->pointlen < gp->n_points) {
surf->points = gf_realloc(surf->points, sizeof(EVG_Vector) * gp->n_points);
if (surf->points == NULL) {
surf->pointlen = 0;
return GF_OUT_OF_MEM;
}
surf->pointlen = gp->n_points;
}
surf->ftoutline.points = surf->points;
for (i=0; i<gp->n_points; i++) {
pt = gp->points[i];
gf_mx2d_apply_point(&surf->mat, &pt);
#ifdef GPAC_FIXED_POINT
surf->points[i].x = pt.x;
surf->points[i].y = pt.y;
#else
surf->points[i].x = (u32) (pt.x * 0x10000L);
surf->points[i].y = (u32) (pt.y * 0x10000L);
#endif
}
#endif
return GF_OK;
}
GF_Err evg_surface_fill(GF_SURFACE _this, GF_STENCIL stencil)
{
GF_Rect rc;
GF_Matrix2D mat, st_mat;
Bool restore_filter;
EVGSurface *surf = (EVGSurface *)_this;
EVGStencil *sten = (EVGStencil *)stencil;
if (!surf || !stencil) return GF_BAD_PARAM;
if (!surf->ftoutline.n_points) return GF_OK;
surf->sten = sten;
if (!setup_grey_callback(surf)) return GF_OK;
get_surface_world_matrix(surf, &mat);
restore_filter = GF_FALSE;
if (sten->type != GF_STENCIL_SOLID) {
rc = surf->path_bounds;
gf_mx2d_apply_rect(&mat, &rc);
rc.x = rc.y = 0;
sten->frame = rc;
gf_mx2d_copy(sten->pmat, surf->mat);
gf_mx2d_inverse(&sten->pmat);
gf_mx2d_copy(st_mat, sten->smat);
gf_mx2d_init(sten->smat);
switch (sten->type) {
case GF_STENCIL_TEXTURE:
if (! ((EVG_Texture *)sten)->pixels) return GF_BAD_PARAM;
if (((EVG_Texture *)sten)->mod & GF_TEXTURE_FLIP) {
if (!surf->center_coords) gf_mx2d_add_scale(&sten->smat, FIX_ONE, -FIX_ONE);
} else {
if (surf->center_coords) gf_mx2d_add_scale(&sten->smat, FIX_ONE, -FIX_ONE);
}
evg_set_texture_active(sten);
gf_mx2d_add_matrix(&sten->smat, &st_mat);
gf_mx2d_add_matrix(&sten->smat, &mat);
gf_mx2d_inverse(&sten->smat);
evg_bmp_init(sten);
if (((EVG_Texture *)sten)->filter == GF_TEXTURE_FILTER_DEFAULT) {
restore_filter = GF_TRUE;
((EVG_Texture *)sten)->filter = surf->texture_filter;
}
break;
case GF_STENCIL_LINEAR_GRADIENT:
{
EVG_LinearGradient *lin = (EVG_LinearGradient *)sten;
gf_mx2d_add_matrix(&sten->smat, &st_mat);
gf_mx2d_add_matrix(&sten->smat, &mat);
gf_mx2d_inverse(&sten->smat);
gf_mx2d_add_matrix(&sten->smat, &lin->vecmat);
gf_mx2d_add_scale(&sten->smat, INT2FIX(1<<EVGGRADIENTBITS), INT2FIX(1<<EVGGRADIENTBITS));
}
break;
case GF_STENCIL_RADIAL_GRADIENT:
{
EVG_RadialGradient *rad = (EVG_RadialGradient*)sten;
gf_mx2d_copy(sten->smat, st_mat);
gf_mx2d_add_matrix(&sten->smat, &mat);
gf_mx2d_inverse(&sten->smat);
gf_mx2d_add_translation(&sten->smat, -rad->center.x, -rad->center.y);
gf_mx2d_add_scale(&sten->smat, gf_invfix(rad->radius.x), gf_invfix(rad->radius.y));
rad->d_f.x = gf_divfix(rad->focus.x - rad->center.x, rad->radius.x);
rad->d_f.y = gf_divfix(rad->focus.y - rad->center.y, rad->radius.y);
evg_radial_init(rad);
}
break;
}
}
if (surf->useClipper) {
surf->ftparams.clip_xMin = surf->clipper.x;
surf->ftparams.clip_yMin = surf->clipper.y;
surf->ftparams.clip_xMax = (surf->clipper.x + surf->clipper.width);
surf->ftparams.clip_yMax = (surf->clipper.y + surf->clipper.height);
} else {
surf->ftparams.clip_xMin = 0;
surf->ftparams.clip_yMin = 0;
surf->ftparams.clip_xMax = (surf->width);
surf->ftparams.clip_yMax = (surf->height);
}
evg_raster_render(surf->raster, &surf->ftparams);
if (sten->type != GF_STENCIL_SOLID) {
gf_mx2d_copy(sten->smat, st_mat);
if (restore_filter) ((EVG_Texture *)sten)->filter = GF_TEXTURE_FILTER_DEFAULT;
}
surf->sten = 0L;
return GF_OK;
}