forked from enlightenment/efl
464 lines
13 KiB
C
464 lines
13 KiB
C
#include "evas_common_private.h"
|
|
#include "evas_engine.h"
|
|
#include <sys/time.h>
|
|
#include <sys/utsname.h>
|
|
|
|
void
|
|
evas_fb_outbuf_fb_init(void)
|
|
{
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_free(Outbuf *buf)
|
|
{
|
|
if (buf->priv.back_buf)
|
|
evas_cache_image_drop(&buf->priv.back_buf->cache_entry);
|
|
fb_freemode(buf->priv.fb.fb);
|
|
fb_cleanup();
|
|
free(buf);
|
|
}
|
|
|
|
static int
|
|
_outbuf_depth_convert(const Outbuf_Depth depth)
|
|
{
|
|
if (depth == OUTBUF_DEPTH_RGB_16BPP_565_565_DITHERED) return 16;
|
|
else if (depth == OUTBUF_DEPTH_RGB_16BPP_555_555_DITHERED) return 15;
|
|
else if (depth == OUTBUF_DEPTH_RGB_16BPP_565_444_DITHERED) return 16;
|
|
else if (depth == OUTBUF_DEPTH_RGB_16BPP_444_444_DITHERED) return 12;
|
|
else if (depth == OUTBUF_DEPTH_RGB_32BPP_888_8888) return 32;
|
|
else if (depth == OUTBUF_DEPTH_INHERIT) return 0;
|
|
return -1;
|
|
}
|
|
|
|
static unsigned int
|
|
fb_bitfield_mask_get(const struct fb_bitfield *fbb)
|
|
{
|
|
unsigned int i, mask = 0;
|
|
for (i = 0; i < fbb->length; i++)
|
|
mask |= (1 << (fbb->offset + i));
|
|
return mask;
|
|
}
|
|
|
|
static Eina_Bool
|
|
_outbuf_reset(Outbuf *buf, int rot, Outbuf_Depth depth)
|
|
{
|
|
Gfx_Func_Convert conv_func = NULL;
|
|
|
|
if (rot == 0 || rot == 180)
|
|
{
|
|
buf->w = buf->priv.fb.fb->width;
|
|
buf->h = buf->priv.fb.fb->height;
|
|
}
|
|
else if (rot == 90 || rot == 270)
|
|
{
|
|
buf->w = buf->priv.fb.fb->height;
|
|
buf->h = buf->priv.fb.fb->width;
|
|
}
|
|
|
|
buf->depth = depth;
|
|
buf->rot = rot;
|
|
|
|
buf->priv.mask.r = fb_bitfield_mask_get(&(buf->priv.fb.fb->fb_var.red));
|
|
buf->priv.mask.g = fb_bitfield_mask_get(&(buf->priv.fb.fb->fb_var.green));
|
|
buf->priv.mask.b = fb_bitfield_mask_get(&(buf->priv.fb.fb->fb_var.blue));
|
|
|
|
conv_func = evas_common_convert_func_get(0,
|
|
buf->priv.fb.fb->width,
|
|
buf->priv.fb.fb->height,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r,
|
|
buf->priv.mask.g,
|
|
buf->priv.mask.b,
|
|
PAL_MODE_NONE,
|
|
buf->rot);
|
|
|
|
DBG("size=%ux%u rot=%u depth=%u bitdepth=%u fb{"
|
|
"width=%u, height=%u, refresh=%u, depth=%u, bpp=%u, fd=%d, "
|
|
"mem=%p, mem_offset=%u, stride=%u pixels} "
|
|
"mask{r=%#010x, g=%#010x, b=%#010x} conv_func=%p",
|
|
buf->w, buf->h, buf->rot, buf->depth,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.fb.fb->width,
|
|
buf->priv.fb.fb->height,
|
|
buf->priv.fb.fb->refresh,
|
|
buf->priv.fb.fb->depth,
|
|
buf->priv.fb.fb->bpp,
|
|
buf->priv.fb.fb->fb_fd,
|
|
buf->priv.fb.fb->mem,
|
|
buf->priv.fb.fb->mem_offset,
|
|
buf->priv.fb.fb->stride,
|
|
buf->priv.mask.r, buf->priv.mask.g, buf->priv.mask.b,
|
|
conv_func);
|
|
|
|
return !!conv_func;
|
|
}
|
|
|
|
Outbuf *
|
|
evas_fb_outbuf_fb_setup_fb(int w, int h, int rot, Outbuf_Depth depth, int vt_no, int dev_no, int refresh)
|
|
{
|
|
/* create outbuf struct */
|
|
/* setup window and/or fb */
|
|
/* if (dithered) create backbuf */
|
|
Outbuf *buf;
|
|
int fb_fd;
|
|
int fb_depth = _outbuf_depth_convert(depth);
|
|
|
|
buf = calloc(1, sizeof(Outbuf));
|
|
if (!buf)
|
|
return NULL;
|
|
|
|
fb_init(vt_no, dev_no);
|
|
if (rot == 0 || rot == 180)
|
|
buf->priv.fb.fb = fb_setmode(w, h, fb_depth, refresh);
|
|
else if (rot == 90 || rot == 270)
|
|
buf->priv.fb.fb = fb_setmode(h, w, fb_depth, refresh);
|
|
if (!buf->priv.fb.fb) buf->priv.fb.fb = fb_getmode();
|
|
if (!buf->priv.fb.fb)
|
|
{
|
|
free(buf);
|
|
return NULL;
|
|
}
|
|
fb_fd = fb_postinit(buf->priv.fb.fb);
|
|
DBG("fd=%d, mode=%ux%u, refresh=%u, depth=%u, bpp=%u, mem=%p, "
|
|
"mem_offset=%u, stride=%u pixels",
|
|
fb_fd, buf->priv.fb.fb->width, buf->priv.fb.fb->height,
|
|
buf->priv.fb.fb->refresh, buf->priv.fb.fb->depth,
|
|
buf->priv.fb.fb->bpp, buf->priv.fb.fb->mem,
|
|
buf->priv.fb.fb->mem_offset, buf->priv.fb.fb->stride);
|
|
if (fb_fd < 1)
|
|
{
|
|
fb_freemode(buf->priv.fb.fb);
|
|
free(buf);
|
|
return NULL;
|
|
}
|
|
|
|
if (!_outbuf_reset(buf, rot, depth))
|
|
{
|
|
fb_freemode(buf->priv.fb.fb);
|
|
fb_cleanup();
|
|
free(buf);
|
|
return NULL;
|
|
}
|
|
|
|
// if (buf->priv.fb.fb->fb_var.bits_per_pixel < 24)
|
|
// buf->priv.back_buf = evas_common_image_create(buf->w, buf->h);
|
|
|
|
return buf;
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_blit(Outbuf *buf, int src_x, int src_y, int w, int h, int dst_x, int dst_y)
|
|
{
|
|
if (buf->priv.back_buf)
|
|
{
|
|
evas_common_blit_rectangle(buf->priv.back_buf, buf->priv.back_buf,
|
|
src_x, src_y, w, h, dst_x, dst_y);
|
|
evas_fb_outbuf_fb_update(buf, dst_x, dst_y, w, h);
|
|
}
|
|
else
|
|
{
|
|
if (buf->priv.fb.fb)
|
|
{
|
|
/* FIXME: need to implement an fb call for "copy area" */
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_update(Outbuf *buf, int x, int y, int w, int h)
|
|
{
|
|
if (!(buf->priv.back_buf)) return;
|
|
if (buf->priv.fb.fb)
|
|
{
|
|
Gfx_Func_Convert conv_func;
|
|
DATA8 *data;
|
|
|
|
data = NULL;
|
|
conv_func = NULL;
|
|
if (buf->rot == 0)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem + buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(x + (y * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, w, h, buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 180)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem + buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(buf->w - x - w + ((buf->h - y - h) * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, w, h, buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 270)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem + buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(buf->h - y - h + (x * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, h, w, buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 90)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem + buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(y + ((buf->w - x - w) * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, h, w, buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
if (conv_func)
|
|
{
|
|
DATA32 *src_data;
|
|
|
|
src_data = buf->priv.back_buf->image.data + (y * buf->w) + x;
|
|
if (buf->rot == 0 || buf->rot == 180)
|
|
{
|
|
conv_func(src_data, data,
|
|
buf->w - w,
|
|
buf->priv.fb.fb->stride - w,
|
|
w, h,
|
|
x, y, NULL);
|
|
}
|
|
else if (buf->rot == 90 || buf->rot == 270)
|
|
{
|
|
conv_func(src_data, data,
|
|
buf->w - w,
|
|
buf->priv.fb.fb->stride - h,
|
|
h, w,
|
|
x, y, NULL);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
RGBA_Image *
|
|
evas_fb_outbuf_fb_new_region_for_update(Outbuf *buf, int x, int y, int w, int h, int *cx, int *cy, int *cw, int *ch)
|
|
{
|
|
if (buf->priv.back_buf)
|
|
{
|
|
*cx = x; *cy = y; *cw = w; *ch = h;
|
|
return buf->priv.back_buf;
|
|
}
|
|
else
|
|
{
|
|
RGBA_Image *im;
|
|
|
|
*cx = 0; *cy = 0; *cw = w; *ch = h;
|
|
im = (RGBA_Image *) evas_cache_image_empty(evas_common_image_cache_get());
|
|
im->cache_entry.flags.alpha = 1;
|
|
im = (RGBA_Image *) evas_cache_image_size_set(&im->cache_entry, w, h);
|
|
|
|
return im;
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_free_region_for_update(Outbuf *buf, RGBA_Image *update)
|
|
{
|
|
if (update != buf->priv.back_buf) evas_cache_image_drop(&update->cache_entry);
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_push_updated_region(Outbuf *buf, RGBA_Image *update, int x, int y, int w, int h)
|
|
{
|
|
if (!buf->priv.fb.fb) return;
|
|
if (buf->priv.back_buf)
|
|
{
|
|
if (update != buf->priv.back_buf)
|
|
evas_common_blit_rectangle(update, buf->priv.back_buf,
|
|
0, 0, w, h, x, y);
|
|
evas_fb_outbuf_fb_update(buf, x, y, w, h);
|
|
}
|
|
else
|
|
{
|
|
Gfx_Func_Convert conv_func;
|
|
DATA8 *data;
|
|
|
|
data = NULL;
|
|
conv_func = NULL;
|
|
if (buf->rot == 0)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem +
|
|
buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(x + (y * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, w, h,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 180)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem +
|
|
buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(buf->w - x - w + ((buf->h - y - h) * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, w, h,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 270)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem +
|
|
buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(buf->h - y - h + (x * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, h, w,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
else if (buf->rot == 90)
|
|
{
|
|
data = (DATA8 *)buf->priv.fb.fb->mem +
|
|
buf->priv.fb.fb->mem_offset +
|
|
buf->priv.fb.fb->bpp *
|
|
(y + ((buf->w - x - w) * buf->priv.fb.fb->stride));
|
|
conv_func = evas_common_convert_func_get(data, h, w,
|
|
buf->priv.fb.fb->fb_var.bits_per_pixel,
|
|
buf->priv.mask.r, buf->priv.mask.g,
|
|
buf->priv.mask.b, PAL_MODE_NONE,
|
|
buf->rot);
|
|
}
|
|
if (conv_func)
|
|
{
|
|
DATA32 *src_data;
|
|
|
|
src_data = update->image.data;
|
|
if (buf->rot == 0 || buf->rot == 180)
|
|
{
|
|
conv_func(src_data, data,
|
|
0,
|
|
buf->priv.fb.fb->stride - w,
|
|
w, h,
|
|
x, y, NULL);
|
|
}
|
|
else if (buf->rot == 90 || buf->rot == 270)
|
|
{
|
|
conv_func(src_data, data,
|
|
0,
|
|
buf->priv.fb.fb->stride - h,
|
|
h, w,
|
|
x, y, NULL);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_reconfigure(Outbuf *buf, int w, int h, int rot, Outbuf_Depth depth)
|
|
{
|
|
int have_backbuf = 0;
|
|
int fb_w, fb_h, fb_depth, refresh;
|
|
|
|
if ((w == buf->w) && (h == buf->h) &&
|
|
(rot == buf->rot) && (depth == buf->depth))
|
|
return;
|
|
if (buf->priv.back_buf)
|
|
{
|
|
evas_cache_image_drop(&buf->priv.back_buf->cache_entry);
|
|
buf->priv.back_buf = NULL;
|
|
have_backbuf = 1;
|
|
}
|
|
|
|
fb_depth = _outbuf_depth_convert(depth);
|
|
refresh = buf->priv.fb.fb->refresh;
|
|
|
|
if (rot == 0 || rot == 180)
|
|
{
|
|
fb_w = w;
|
|
fb_h = h;
|
|
}
|
|
else
|
|
{
|
|
fb_w = h;
|
|
fb_h = w;
|
|
}
|
|
|
|
if (buf->priv.fb.fb)
|
|
buf->priv.fb.fb = fb_changemode(buf->priv.fb.fb, fb_w, fb_h,
|
|
fb_depth, refresh);
|
|
else
|
|
buf->priv.fb.fb = fb_setmode(fb_w, fb_h, fb_depth, refresh);
|
|
|
|
if (!buf->priv.fb.fb) buf->priv.fb.fb = fb_getmode();
|
|
EINA_SAFETY_ON_NULL_RETURN(buf->priv.fb.fb);
|
|
|
|
if (!_outbuf_reset(buf, rot, depth))
|
|
return;
|
|
|
|
evas_fb_outbuf_fb_set_have_backbuf(buf, have_backbuf);
|
|
|
|
/* if backbuf delet it */
|
|
/* resize window or reset fb mode */
|
|
/* if (dithered) create new backbuf */
|
|
}
|
|
|
|
int
|
|
evas_fb_outbuf_fb_get_width(Outbuf *buf)
|
|
{
|
|
return buf->w;
|
|
}
|
|
|
|
int
|
|
evas_fb_outbuf_fb_get_height(Outbuf *buf)
|
|
{
|
|
return buf->h;
|
|
}
|
|
|
|
Outbuf_Depth
|
|
evas_fb_outbuf_fb_get_depth(Outbuf *buf)
|
|
{
|
|
return buf->depth;
|
|
}
|
|
|
|
int
|
|
evas_fb_outbuf_fb_get_rot(Outbuf *buf)
|
|
{
|
|
return buf->rot;
|
|
}
|
|
|
|
int
|
|
evas_fb_outbuf_fb_get_have_backbuf(Outbuf *buf)
|
|
{
|
|
if (buf->priv.back_buf) return 1;
|
|
return 0;
|
|
}
|
|
|
|
void
|
|
evas_fb_outbuf_fb_set_have_backbuf(Outbuf *buf, int have_backbuf)
|
|
{
|
|
if (buf->priv.back_buf)
|
|
{
|
|
if (have_backbuf) return;
|
|
evas_cache_image_drop(&buf->priv.back_buf->cache_entry);
|
|
buf->priv.back_buf = NULL;
|
|
}
|
|
else
|
|
{
|
|
if (!have_backbuf) return;
|
|
if (buf->priv.fb.fb)
|
|
{
|
|
if (buf->priv.fb.fb->fb_var.bits_per_pixel < 24)
|
|
{
|
|
buf->priv.back_buf = (RGBA_Image *) evas_cache_image_empty(evas_common_image_cache_get());
|
|
buf->priv.back_buf = (RGBA_Image *) evas_cache_image_size_set(&buf->priv.back_buf->cache_entry, buf->w, buf->h);
|
|
}
|
|
}
|
|
}
|
|
}
|