956 lines
24 KiB
C
956 lines
24 KiB
C
#ifdef HAVE_CONFIG_H
|
|
# include <config.h>
|
|
#endif
|
|
|
|
#include "evas_common.h"
|
|
#include "evas_macros.h"
|
|
#include "evas_xlib_swapper.h"
|
|
|
|
#ifdef HAVE_DLSYM
|
|
# include <dlfcn.h> /* dlopen,dlclose,etc */
|
|
# include <sys/types.h>
|
|
# include <sys/stat.h>
|
|
# include <fcntl.h>
|
|
|
|
#if 0
|
|
// X(shm)image emulation of multiple buffers + swapping /////////////////////
|
|
typedef struct
|
|
{
|
|
XImage *xim;
|
|
XShmSegmentInfo shm_info;
|
|
void *data;
|
|
int w, h, bpl;
|
|
Eina_Bool shm : 1;
|
|
Eina_Bool valid : 1;
|
|
} Buffer;
|
|
|
|
struct _X_Swapper
|
|
{
|
|
Display *disp;
|
|
Drawable draw;
|
|
Visual *vis;
|
|
GC gc;
|
|
Buffer buf[3];
|
|
int w, h, depth;
|
|
int buf_cur, buf_num;
|
|
Eina_Bool mapped: 1;
|
|
};
|
|
|
|
static Eina_Bool _x_err = EINA_FALSE;
|
|
|
|
static void
|
|
_x_err_hand(Display *d EINA_UNUSED, XErrorEvent *ev EINA_UNUSED)
|
|
{
|
|
_x_err = 1;
|
|
return;
|
|
}
|
|
|
|
static Eina_Bool
|
|
_buf_new(X_Swapper *swp, Buffer *buf)
|
|
{
|
|
buf->w = swp->w;
|
|
buf->h = swp->h;
|
|
|
|
// try shm first
|
|
buf->xim = XShmCreateImage(swp->disp, swp->vis, swp->depth, ZPixmap,
|
|
NULL, &(buf->shm_info), buf->w, buf->h);
|
|
if (buf->xim)
|
|
{
|
|
buf->bpl = buf->xim->bytes_per_line;
|
|
|
|
buf->shm_info.shmid = shmget(IPC_PRIVATE, buf->bpl * buf->h,
|
|
IPC_CREAT | 0777);
|
|
if (buf->shm_info.shmid >= 0)
|
|
{
|
|
buf->shm_info.readOnly = False;
|
|
buf->shm_info.shmaddr = buf->data = buf->xim->data =
|
|
shmat(buf->shm_info.shmid, 0, 0);
|
|
if (buf->shm_info.shmaddr != ((void *)-1))
|
|
{
|
|
XErrorHandler ph;
|
|
|
|
XSync(swp->disp, False);
|
|
_x_err = EINA_FALSE;
|
|
ph = XSetErrorHandler((XErrorHandler)_x_err_hand);
|
|
XShmAttach(swp->disp, &(buf->shm_info));
|
|
XSync(swp->disp, False);
|
|
XSetErrorHandler((XErrorHandler)ph);
|
|
if (!_x_err)
|
|
{
|
|
buf->shm = EINA_TRUE;
|
|
}
|
|
else
|
|
{
|
|
shmdt(buf->shm_info.shmaddr);
|
|
shmctl(buf->shm_info.shmid, IPC_RMID, 0);
|
|
XDestroyImage(buf->xim);
|
|
buf->xim = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
shmdt(buf->shm_info.shmaddr);
|
|
shmctl(buf->shm_info.shmid, IPC_RMID, 0);
|
|
XDestroyImage(buf->xim);
|
|
buf->xim = NULL;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
XDestroyImage(buf->xim);
|
|
buf->xim = NULL;
|
|
}
|
|
}
|
|
|
|
if (!buf->xim) // shm failed - try normal ximage
|
|
{
|
|
buf->xim = XCreateImage(swp->disp, swp->vis, swp->depth, ZPixmap,
|
|
0, NULL, buf->w, buf->h, 32, 0);
|
|
if (!buf->xim) return EINA_FALSE;
|
|
buf->bpl = buf->xim->bytes_per_line;
|
|
buf->data = buf->xim->data = malloc(buf->bpl * buf->h);
|
|
if (!buf->data)
|
|
{
|
|
XDestroyImage(buf->xim);
|
|
buf->xim = NULL;
|
|
return EINA_FALSE;
|
|
}
|
|
}
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static void
|
|
_buf_free(X_Swapper *swp, Buffer *buf)
|
|
{
|
|
if (!buf->xim) return;
|
|
if (buf->shm)
|
|
{
|
|
XShmDetach(swp->disp, &(buf->shm_info));
|
|
XDestroyImage(buf->xim);
|
|
shmdt(buf->shm_info.shmaddr);
|
|
shmctl(buf->shm_info.shmid, IPC_RMID, 0);
|
|
}
|
|
else
|
|
{
|
|
XDestroyImage(buf->xim);
|
|
}
|
|
buf->xim = NULL;
|
|
buf->shm = EINA_FALSE;
|
|
}
|
|
|
|
static void
|
|
_buf_put(X_Swapper *swp, Buffer *buf, Eina_Rectangle *rects, int nrects)
|
|
{
|
|
Region tmpr;
|
|
int i;
|
|
|
|
if (!buf->xim) return;
|
|
tmpr = XCreateRegion();
|
|
if ((rects)/* && 0*/) // set to 0 to test buffer stuff
|
|
{
|
|
for (i = 0; i < nrects; i++)
|
|
{
|
|
XRectangle xr;
|
|
|
|
xr.x = rects[i].x; xr.y = rects[i].y;
|
|
xr.width = rects[i].w; xr.height = rects[i].h;
|
|
XUnionRectWithRegion(&xr, tmpr, tmpr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
XRectangle xr;
|
|
|
|
xr.x = 0; xr.y = 0;
|
|
xr.width = buf->w; xr.height = buf->h;
|
|
XUnionRectWithRegion(&xr, tmpr, tmpr);
|
|
}
|
|
XSetRegion(swp->disp, swp->gc, tmpr);
|
|
XDestroyRegion(tmpr);
|
|
if (buf->shm)
|
|
{
|
|
XShmPutImage(swp->disp, swp->draw, swp->gc, buf->xim, 0, 0, 0, 0,
|
|
buf->w, buf->h, False);
|
|
}
|
|
else
|
|
{
|
|
XPutImage(swp->disp, swp->draw, swp->gc, buf->xim, 0, 0, 0, 0,
|
|
buf->w, buf->h);
|
|
}
|
|
XSync(swp->disp, False);
|
|
}
|
|
|
|
//////////////////////////////////////////////////////////////////////////////
|
|
|
|
X_Swapper *
|
|
evas_xlib_swapper_new(Display *disp, Drawable draw, Visual *vis,
|
|
int depth, int w, int h)
|
|
{
|
|
X_Swapper *swp;
|
|
XGCValues gcv;
|
|
int i;
|
|
|
|
int nbuf = 3; // pretend we are triple buffered
|
|
|
|
swp = calloc(1, sizeof(X_Swapper));
|
|
if (!swp) return NULL;
|
|
swp->disp = disp;
|
|
swp->draw = draw;
|
|
swp->vis = vis;
|
|
swp->depth = depth;
|
|
swp->w = w;
|
|
swp->h = h;
|
|
swp->buf_num = nbuf;
|
|
swp->gc = XCreateGC(swp->disp, swp->draw, 0, &gcv);
|
|
for (i = 0; i < swp->buf_num; i++)
|
|
{
|
|
if (!_buf_new(swp, &(swp->buf[i])))
|
|
{
|
|
evas_xlib_swapper_free(swp);
|
|
return NULL;
|
|
}
|
|
}
|
|
return swp;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_free(X_Swapper *swp)
|
|
{
|
|
int i;
|
|
|
|
for (i = 0; i < swp->buf_num; i++)
|
|
{
|
|
_buf_free(swp, &(swp->buf[i]));
|
|
}
|
|
XFreeGC(swp->disp, swp->gc);
|
|
free(swp);
|
|
}
|
|
|
|
void *
|
|
evas_xlib_swapper_buffer_map(X_Swapper *swp, int *bpl, int *w, int *h)
|
|
{
|
|
swp->mapped = EINA_TRUE;
|
|
if (bpl) *bpl = swp->buf[swp->buf_cur].bpl;
|
|
if (w) *w = swp->w;
|
|
if (h) *h = swp->h;
|
|
return swp->buf[swp->buf_cur].data;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_buffer_unmap(X_Swapper *swp)
|
|
{
|
|
swp->mapped = EINA_FALSE;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_swap(X_Swapper *swp, Eina_Rectangle *rects, int nrects)
|
|
{
|
|
int n;
|
|
|
|
n = swp->buf_cur;
|
|
_buf_put(swp, &(swp->buf[n]), rects, nrects);
|
|
swp->buf[n].valid = 1;
|
|
swp->buf_cur = (swp->buf_cur + 1) % swp->buf_num;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_buffer_state_get(X_Swapper *swp)
|
|
{
|
|
int i, n, count = 0;
|
|
/*
|
|
for (i = 0; i < swp->buf_num; i++)
|
|
{
|
|
if ((rand() % 50) == 0)
|
|
swp->buf[n].valid = 0;
|
|
}
|
|
*/
|
|
for (i = 0; i < swp->buf_num; i++)
|
|
{
|
|
n = (swp->buf_num + swp->buf_cur - (i)) % swp->buf_num;
|
|
if (swp->buf[n].valid) count++;
|
|
else break;
|
|
}
|
|
if (count == swp->buf_num)
|
|
{
|
|
if (count == 1) return MODE_COPY;
|
|
else if (count == 2) return MODE_DOUBLE;
|
|
else if (count == 3) return MODE_TRIPLE;
|
|
}
|
|
return MODE_FULL;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_depth_get(X_Swapper *swp)
|
|
{
|
|
return swp->buf[0].xim->bits_per_pixel;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_byte_order_get(X_Swapper *swp)
|
|
{
|
|
return swp->buf[0].xim->byte_order;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_bit_order_get(X_Swapper *swp)
|
|
{
|
|
return swp->buf[0].xim->bitmap_bit_order;
|
|
}
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// DRM/DRI buffer swapping+access (driver specific) /////////////////////
|
|
|
|
static Eina_Bool tried = EINA_FALSE;
|
|
////////////////////////////////////
|
|
//libdrm.so.2
|
|
static void *drm_lib = NULL;
|
|
|
|
typedef unsigned int drm_magic_t;
|
|
static int (*sym_drmGetMagic) (int fd, drm_magic_t *magic) = NULL;
|
|
|
|
////////////////////////////////////
|
|
// libdrm_slp.so.1
|
|
#define DRM_SLP_DEVICE_CPU 1
|
|
#define DRM_SLP_OPTION_READ (1 << 0)
|
|
#define DRM_SLP_OPTION_WRITE (1 << 1)
|
|
static void *drm_slp_lib = NULL;
|
|
|
|
typedef struct _drm_slp_bufmgr *drm_slp_bufmgr;
|
|
typedef struct _drm_slp_bo *drm_slp_bo;
|
|
static drm_slp_bo (*sym_drm_slp_bo_import) (drm_slp_bufmgr bufmgr, unsigned int key) = NULL;
|
|
// XXXX: sym_drm_slp_bo_map() is incorrectly defined - it SHOULD return a
|
|
// void * at least
|
|
static void *(*sym_drm_slp_bo_map) (drm_slp_bo bo, int device, int opt) = NULL;
|
|
static int (*sym_drm_slp_bo_unmap) (drm_slp_bo bo, int device) = NULL;
|
|
static void (*sym_drm_slp_bo_unref) (drm_slp_bo bo) = NULL;
|
|
static drm_slp_bufmgr (*sym_drm_slp_bufmgr_init) (int fd, void *arg) = NULL;
|
|
static void (*sym_drm_slp_bufmgr_destroy) (drm_slp_bufmgr bufmgr) = NULL;
|
|
|
|
////////////////////////////////////
|
|
// libdri2.so.0
|
|
#define DRI2BufferBackLeft 1
|
|
static void *dri_lib = NULL;
|
|
|
|
typedef unsigned long long CD64;
|
|
|
|
typedef struct
|
|
{
|
|
unsigned int attachment;
|
|
unsigned int name;
|
|
unsigned int pitch;
|
|
unsigned int cpp;
|
|
unsigned int flags;
|
|
} DRI2Buffer;
|
|
|
|
#define DRI2_BUFFER_TYPE_WINDOW 0x0
|
|
#define DRI2_BUFFER_TYPE_PIXMAP 0x1
|
|
#define DRI2_BUFFER_TYPE_FB 0x2
|
|
|
|
typedef union
|
|
{
|
|
unsigned int flags;
|
|
struct
|
|
{
|
|
unsigned int type:1;
|
|
unsigned int is_framebuffer:1;
|
|
unsigned int is_mapped:1;
|
|
unsigned int is_reused:1;
|
|
unsigned int idx_reuse:3;
|
|
}
|
|
data;
|
|
} DRI2BufferFlags;
|
|
|
|
static DRI2Buffer *(*sym_DRI2GetBuffers) (Display *display, XID drawable, int *width, int *height, unsigned int *attachments, int count, int *outCount) = NULL;
|
|
static Bool (*sym_DRI2QueryExtension) (Display *display, int *eventBase, int *errorBase) = NULL;
|
|
static Bool (*sym_DRI2QueryVersion) (Display *display, int *major, int *minor) = NULL;
|
|
static Bool (*sym_DRI2Connect) (Display *display, XID window, char **driverName, char **deviceName) = NULL;
|
|
static Bool (*sym_DRI2Authenticate) (Display *display, XID window, unsigned int magic) = NULL;
|
|
static void (*sym_DRI2CreateDrawable) (Display *display, XID drawable) = NULL;
|
|
static void (*sym_DRI2SwapBuffersWithRegion) (Display *display, XID drawable, XID region, CD64 *count) = NULL;
|
|
static void (*sym_DRI2SwapBuffers) (Display *display, XID drawable, CD64 target_msc, CD64 divisor, CD64 remainder, CD64 *count) = NULL;
|
|
static void (*sym_DRI2DestroyDrawable) (Display *display, XID handle) = NULL;
|
|
|
|
////////////////////////////////////
|
|
// libXfixes.so.3
|
|
static void *xfixes_lib = NULL;
|
|
|
|
static Bool (*sym_XFixesQueryExtension) (Display *display, int *event_base_return, int *error_base_return) = NULL;
|
|
static Status (*sym_XFixesQueryVersion) (Display *display, int *major_version_return, int *minor_version_return) = NULL;
|
|
static XID (*sym_XFixesCreateRegion) (Display *display, XRectangle *rectangles, int nrectangles) = NULL;
|
|
static void (*sym_XFixesDestroyRegion) (Display *dpy, XID region) = NULL;
|
|
|
|
////////////////////////////////////////////////////////////////////////////
|
|
#define MAX_BO_CACHE 4
|
|
|
|
typedef struct
|
|
{
|
|
unsigned int name;
|
|
drm_slp_bo buf_bo;
|
|
} Buffer;
|
|
|
|
struct _X_Swapper
|
|
{
|
|
Display *disp;
|
|
Drawable draw;
|
|
Visual *vis;
|
|
int w, h, depth;
|
|
drm_slp_bo buf_bo;
|
|
DRI2Buffer *buf;
|
|
void *buf_data;
|
|
int buf_w, buf_h;
|
|
Eina_List *buf_cache;
|
|
int last_count;
|
|
Eina_Bool mapped: 1;
|
|
};
|
|
|
|
static int inits = 0;
|
|
static int xfixes_ev_base = 0, xfixes_err_base = 0;
|
|
static int xfixes_major = 0, xfixes_minor = 0;
|
|
static int dri2_ev_base = 0, dri2_err_base = 0;
|
|
static int dri2_major = 0, dri2_minor = 0;
|
|
static int drm_fd = -1;
|
|
static drm_slp_bufmgr bufmgr = NULL;
|
|
static int swap_debug = -1;
|
|
|
|
static Eina_Bool
|
|
_drm_init(Display *disp, int scr)
|
|
{
|
|
char *drv_name = NULL, *dev_name = NULL;
|
|
drm_magic_t magic = 0;
|
|
|
|
if (swap_debug == -1)
|
|
{
|
|
if (getenv("EVAS_SWAPPER_DEBUG")) swap_debug = 1;
|
|
else swap_debug = 0;
|
|
}
|
|
|
|
if (xfixes_lib) return EINA_TRUE;
|
|
if ((tried) && (!xfixes_lib)) return EINA_FALSE;
|
|
tried = EINA_TRUE;
|
|
drm_lib = dlopen("libdrm.so.2", RTLD_NOW | RTLD_LOCAL);
|
|
if (!drm_lib)
|
|
{
|
|
if (swap_debug) ERR("Can't load libdrm.so.2");
|
|
goto err;
|
|
}
|
|
drm_slp_lib = dlopen("libdrm_slp.so.1", RTLD_NOW | RTLD_LOCAL);
|
|
if (!drm_slp_lib)
|
|
{
|
|
if (swap_debug) ERR("Can't load libdrm_slp.so.1");
|
|
goto err;
|
|
}
|
|
dri_lib = dlopen("libdri2.so.0", RTLD_NOW | RTLD_LOCAL);
|
|
if (!dri_lib)
|
|
{
|
|
if (swap_debug) ERR("Can't load libdri2.so.0");
|
|
goto err;
|
|
}
|
|
xfixes_lib = dlopen("libXfixes.so.3", RTLD_NOW | RTLD_LOCAL);
|
|
if (!xfixes_lib)
|
|
{
|
|
if (swap_debug) ERR("Can't load libXfixes.so.3");
|
|
goto err;
|
|
}
|
|
|
|
#define SYM(l, x) \
|
|
do { sym_ ## x = dlsym(l, #x); \
|
|
if (!sym_ ## x) { \
|
|
if (swap_debug) ERR("Can't load symbol "#x); \
|
|
goto err; \
|
|
} \
|
|
} while (0)
|
|
|
|
SYM(drm_lib, drmGetMagic);
|
|
|
|
SYM(drm_slp_lib, drm_slp_bo_import);
|
|
SYM(drm_slp_lib, drm_slp_bo_map);
|
|
SYM(drm_slp_lib, drm_slp_bo_unmap);
|
|
SYM(drm_slp_lib, drm_slp_bo_unref);
|
|
SYM(drm_slp_lib, drm_slp_bufmgr_init);
|
|
SYM(drm_slp_lib, drm_slp_bufmgr_destroy);
|
|
|
|
SYM(dri_lib, DRI2GetBuffers);
|
|
SYM(dri_lib, DRI2QueryExtension);
|
|
SYM(dri_lib, DRI2QueryVersion);
|
|
SYM(dri_lib, DRI2Connect);
|
|
SYM(dri_lib, DRI2Authenticate);
|
|
SYM(dri_lib, DRI2CreateDrawable);
|
|
SYM(dri_lib, DRI2SwapBuffersWithRegion);
|
|
SYM(dri_lib, DRI2SwapBuffers);
|
|
SYM(dri_lib, DRI2DestroyDrawable);
|
|
|
|
SYM(xfixes_lib, XFixesQueryExtension);
|
|
SYM(xfixes_lib, XFixesQueryVersion);
|
|
SYM(xfixes_lib, XFixesCreateRegion);
|
|
SYM(xfixes_lib, XFixesDestroyRegion);
|
|
|
|
if (!sym_XFixesQueryExtension(disp, &xfixes_ev_base, &xfixes_err_base))
|
|
{
|
|
if (swap_debug) ERR("XFixes extension not in xserver");
|
|
goto err;
|
|
}
|
|
sym_XFixesQueryVersion(disp, &xfixes_major, &xfixes_minor);
|
|
|
|
if (!sym_DRI2QueryExtension(disp, &dri2_ev_base, &dri2_err_base))
|
|
{
|
|
if (swap_debug) ERR("DRI2 extension not in xserver");
|
|
goto err;
|
|
}
|
|
if (!sym_DRI2QueryVersion(disp, &dri2_major, &dri2_minor))
|
|
{
|
|
if (swap_debug) ERR("DRI2 query version failed");
|
|
goto err;
|
|
}
|
|
if (dri2_minor < 99)
|
|
{
|
|
if (swap_debug) ERR("Not supported by DRI2 version(%i.%i)",
|
|
dri2_major, dri2_minor);
|
|
goto err;
|
|
}
|
|
if (!sym_DRI2Connect(disp, RootWindow(disp, scr), &drv_name, &dev_name))
|
|
{
|
|
if (swap_debug) ERR("DRI2 connect failed on screen %i", scr);
|
|
goto err;
|
|
}
|
|
drm_fd = open(dev_name, O_RDWR);
|
|
if (drm_fd < 0)
|
|
{
|
|
if (swap_debug) ERR("DRM FD open of '%s' failed", dev_name);
|
|
goto err;
|
|
}
|
|
if (sym_drmGetMagic(drm_fd, &magic))
|
|
{
|
|
if (swap_debug) ERR("DRM get magic failed");
|
|
goto err;
|
|
}
|
|
if (!sym_DRI2Authenticate(disp, RootWindow(disp, scr),
|
|
(unsigned int)magic))
|
|
{
|
|
if (swap_debug) ERR("DRI2 authenticate failed with magic 0x%x on screen %i", (unsigned int)magic, scr);
|
|
goto err;
|
|
}
|
|
|
|
if (!(bufmgr = sym_drm_slp_bufmgr_init(drm_fd, NULL)))
|
|
{
|
|
if (swap_debug) ERR("DRM bufmgr init failed");
|
|
goto err;
|
|
}
|
|
return EINA_TRUE;
|
|
err:
|
|
if (drm_fd >= 0)
|
|
{
|
|
close(drm_fd);
|
|
drm_fd = -1;
|
|
}
|
|
if (drm_lib)
|
|
{
|
|
dlclose(drm_lib);
|
|
drm_lib = NULL;
|
|
}
|
|
if (drm_slp_lib)
|
|
{
|
|
dlclose(drm_slp_lib);
|
|
drm_slp_lib = NULL;
|
|
}
|
|
if (dri_lib)
|
|
{
|
|
dlclose(dri_lib);
|
|
dri_lib = NULL;
|
|
}
|
|
if (xfixes_lib)
|
|
{
|
|
dlclose(xfixes_lib);
|
|
xfixes_lib = NULL;
|
|
}
|
|
return EINA_FALSE;
|
|
}
|
|
|
|
static void
|
|
_drm_shutdown(void)
|
|
{
|
|
return;
|
|
// leave this here as notation on how to shut down stuff - never do it
|
|
// though, as once shut down, we have to re-init and this could be
|
|
// expensive especially if u have a single canvas that is changing config
|
|
// and being shut down and re-initted a few times.
|
|
if (bufmgr)
|
|
{
|
|
sym_drm_slp_bufmgr_destroy(bufmgr);
|
|
bufmgr = NULL;
|
|
}
|
|
if (drm_fd >= 0) close(drm_fd);
|
|
drm_fd = -1;
|
|
dlclose(drm_slp_lib);
|
|
drm_slp_lib = NULL;
|
|
dlclose(dri_lib);
|
|
dri_lib = NULL;
|
|
dlclose(xfixes_lib);
|
|
xfixes_lib = NULL;
|
|
}
|
|
|
|
static Eina_Bool
|
|
_drm_setup(X_Swapper *swp)
|
|
{
|
|
sym_DRI2CreateDrawable(swp->disp, swp->draw);
|
|
return EINA_TRUE;
|
|
}
|
|
|
|
static void
|
|
_drm_cleanup(X_Swapper *swp)
|
|
{
|
|
sym_DRI2DestroyDrawable(swp->disp, swp->draw);
|
|
}
|
|
|
|
X_Swapper *
|
|
evas_xlib_swapper_new(Display *disp, Drawable draw, Visual *vis,
|
|
int depth, int w, int h)
|
|
{
|
|
X_Swapper *swp;
|
|
|
|
if (inits <= 0)
|
|
{
|
|
if (!_drm_init(disp, 0)) return NULL;
|
|
}
|
|
inits++;
|
|
|
|
swp = calloc(1, sizeof(X_Swapper));
|
|
if (!swp) return NULL;
|
|
swp->disp = disp;
|
|
swp->draw = draw;
|
|
swp->vis = vis;
|
|
swp->depth = depth;
|
|
swp->w = w;
|
|
swp->h = h;
|
|
swp->last_count = -1;
|
|
if (!_drm_setup(swp))
|
|
{
|
|
inits--;
|
|
if (inits == 0) _drm_shutdown();
|
|
free(swp);
|
|
return NULL;
|
|
}
|
|
if (swp->depth == 24)
|
|
{ // need to adjust to 32bpp?? have to check
|
|
swp->depth = 32;
|
|
}
|
|
if (swap_debug) printf("Swapper allocated OK\n");
|
|
return swp;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_free(X_Swapper *swp)
|
|
{
|
|
Buffer *b;
|
|
|
|
if (swap_debug) printf("Swapper free\n");
|
|
if (swp->mapped) evas_xlib_swapper_buffer_unmap(swp);
|
|
EINA_LIST_FREE(swp->buf_cache, b)
|
|
{
|
|
if (swap_debug) printf("Cached buf name %i freed\n", b->name);
|
|
sym_drm_slp_bo_unref(b->buf_bo);
|
|
free(b);
|
|
}
|
|
_drm_cleanup(swp);
|
|
free(swp);
|
|
inits--;
|
|
if (inits == 0) _drm_shutdown();
|
|
}
|
|
|
|
void *
|
|
evas_xlib_swapper_buffer_map(X_Swapper *swp, int *bpl, int *w, int *h)
|
|
{
|
|
unsigned int attach = DRI2BufferBackLeft;
|
|
int num;
|
|
Eina_List *l;
|
|
Buffer *b;
|
|
DRI2BufferFlags *flags;
|
|
|
|
if (swp->mapped)
|
|
{
|
|
if (bpl)
|
|
{
|
|
if ((swp->buf) && (swp->buf->pitch > 0)) *bpl = swp->buf->pitch;
|
|
else *bpl = swp->w * 4;
|
|
}
|
|
return swp->buf_data;
|
|
}
|
|
swp->buf = sym_DRI2GetBuffers(swp->disp, swp->draw,
|
|
&(swp->buf_w), &(swp->buf_h),
|
|
&attach, 1, &num);
|
|
if (!swp->buf) return NULL;
|
|
if (!swp->buf->name) return NULL;
|
|
flags = (DRI2BufferFlags *)(&(swp->buf->flags));
|
|
if (!flags->data.is_reused)
|
|
{
|
|
if (swap_debug) printf("Buffer cache not reused - clear cache\n");
|
|
// buffer isnt recycled - nuke the buf cache
|
|
EINA_LIST_FREE(swp->buf_cache, b)
|
|
{
|
|
if (swap_debug) printf("Cached buf name %i freed\n", b->name);
|
|
sym_drm_slp_bo_unref(b->buf_bo);
|
|
free(b);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// find a cache buf entry
|
|
EINA_LIST_FOREACH(swp->buf_cache, l, b)
|
|
{
|
|
if (b->name == swp->buf->name)
|
|
{
|
|
if (swap_debug) printf("Cached buf name %i found\n", b->name);
|
|
swp->buf_bo = b->buf_bo;
|
|
// LRU - least used at end. found item - promote to front
|
|
swp->buf_cache = eina_list_promote_list(swp->buf_cache, l);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
if (!swp->buf_bo)
|
|
{
|
|
swp->buf_bo = sym_drm_slp_bo_import(bufmgr, swp->buf->name);
|
|
if (!swp->buf_bo) return NULL;
|
|
// cache the buf entry
|
|
b = calloc(1, sizeof(Buffer));
|
|
b->name = swp->buf->name;
|
|
b->buf_bo = swp->buf_bo;
|
|
// put ah head of list
|
|
swp->buf_cache = eina_list_prepend(swp->buf_cache, b);
|
|
if (swap_debug) printf("Buffer cache added name %i\n", b->name);
|
|
// keep bo cache no more than its max size
|
|
while (eina_list_count(swp->buf_cache) > MAX_BO_CACHE)
|
|
{
|
|
if (swap_debug) printf("Buffer cache count %i more than max %i\n", eina_list_count(swp->buf_cache) , MAX_BO_CACHE);
|
|
l = eina_list_last(swp->buf_cache);
|
|
if (l)
|
|
{
|
|
b = l->data;
|
|
if (swap_debug) printf("Buffer cache overfull - free name %i\n", b->name);
|
|
swp->buf_cache = eina_list_remove_list(swp->buf_cache, l);
|
|
sym_drm_slp_bo_unref(b->buf_bo);
|
|
free(b);
|
|
}
|
|
}
|
|
}
|
|
// XXXX: sym_drm_slp_bo_map() is incorrectly defined - it SHOULD return a
|
|
// void * at least
|
|
swp->buf_data = sym_drm_slp_bo_map(swp->buf_bo, DRM_SLP_DEVICE_CPU,
|
|
DRM_SLP_OPTION_READ |
|
|
DRM_SLP_OPTION_WRITE);
|
|
if (!swp->buf_data)
|
|
{
|
|
ERR("Buffer map name %i failed", swp->buf->name);
|
|
return NULL;
|
|
}
|
|
if (bpl) *bpl = swp->buf->pitch;
|
|
swp->mapped = EINA_TRUE;
|
|
if (swap_debug) printf("Mapped bufer name %i OK\n", swp->buf->name);
|
|
if ((swp->w != swp->buf_w) || (swp->h != swp->buf_h))
|
|
{
|
|
ERR("Evas software DRI swapper buffer size mismatch");
|
|
}
|
|
swp->w = swp->buf_w;
|
|
swp->h = swp->buf_h;
|
|
if (w) *w = swp->w;
|
|
if (h) *h = swp->h;
|
|
return swp->buf_data;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_buffer_unmap(X_Swapper *swp)
|
|
{
|
|
if (!swp->mapped) return;
|
|
sym_drm_slp_bo_unmap(swp->buf_bo, DRM_SLP_DEVICE_CPU);
|
|
if (swap_debug) printf("Unmap buffer name %i\n", swp->buf->name);
|
|
free(swp->buf);
|
|
swp->buf = NULL;
|
|
swp->buf_bo = NULL;
|
|
swp->buf_data = NULL;
|
|
swp->mapped = EINA_FALSE;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_swap(X_Swapper *swp, Eina_Rectangle *rects, int nrects)
|
|
{
|
|
XRectangle *xrects = alloca(nrects * sizeof(XRectangle));
|
|
XID region;
|
|
int i;
|
|
unsigned long long sbc_count = 0;
|
|
|
|
if (swap_debug) printf("Swap buffers\n");
|
|
for (i = 0; i < nrects; i++)
|
|
{
|
|
xrects[i].x = rects[i].x; xrects[i].y = rects[i].y;
|
|
xrects[i].width = rects[i].w; xrects[i].height = rects[i].h;
|
|
}
|
|
region = sym_XFixesCreateRegion(swp->disp, xrects, nrects);
|
|
sym_DRI2SwapBuffersWithRegion(swp->disp, swp->draw, region, &sbc_count);
|
|
sym_XFixesDestroyRegion(swp->disp, region);
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_buffer_state_get(X_Swapper *swp)
|
|
{
|
|
DRI2BufferFlags *flags;
|
|
|
|
if (!swp->mapped) evas_xlib_swapper_buffer_map(swp, NULL, NULL, NULL);
|
|
if (!swp->mapped) return MODE_FULL;
|
|
flags = (DRI2BufferFlags *)(&(swp->buf->flags));
|
|
if (flags->data.idx_reuse != swp->last_count)
|
|
{
|
|
static int force_full_on_reuse_change = -1;
|
|
|
|
swp->last_count = flags->data.idx_reuse;
|
|
if (force_full_on_reuse_change == -1)
|
|
{
|
|
if (getenv("EVAS_FORCE_FULL_ON_REUSE_CHANGE")) force_full_on_reuse_change = 1;
|
|
else force_full_on_reuse_change = 0;
|
|
}
|
|
if (force_full_on_reuse_change)
|
|
{
|
|
if (swap_debug) printf("Reuse changed - force FULL\n");
|
|
return MODE_FULL;
|
|
}
|
|
}
|
|
if (swap_debug) printf("Swap state idx_reuse = %i (0=FULL, 1=COPY, 2=DOUBLE, 3=TRIPLE)\n", flags->data.idx_reuse);
|
|
if (flags->data.idx_reuse == 0) return MODE_FULL;
|
|
else if (flags->data.idx_reuse == 1) return MODE_COPY;
|
|
else if (flags->data.idx_reuse == 2) return MODE_DOUBLE;
|
|
else if (flags->data.idx_reuse == 3) return MODE_TRIPLE;
|
|
return MODE_FULL;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_depth_get(X_Swapper *swp)
|
|
{
|
|
return swp->depth;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_byte_order_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return LSBFirst;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_bit_order_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return LSBFirst;
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
X_Swapper *
|
|
evas_xlib_swapper_new(Display *disp EINA_UNUSED, Drawable draw EINA_UNUSED,
|
|
Visual *vis EINA_UNUSED, int depth EINA_UNUSED,
|
|
int w EINA_UNUSED, int h EINA_UNUSED)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_free(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
}
|
|
|
|
void *
|
|
evas_xlib_swapper_buffer_map(X_Swapper *swp EINA_UNUSED, int *bpl EINA_UNUSED, int *w EINA_UNUSED, int *h EINA_UNUSED)
|
|
{
|
|
return NULL;
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_buffer_unmap(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
}
|
|
|
|
void
|
|
evas_xlib_swapper_swap(X_Swapper *swp EINA_UNUSED, Eina_Rectangle *rects EINA_UNUSED, int nrects EINA_UNUSED)
|
|
{
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_buffer_state_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return MODE_FULL;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_depth_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_byte_order_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
int
|
|
evas_xlib_swapper_bit_order_get(X_Swapper *swp EINA_UNUSED)
|
|
{
|
|
return 0;
|
|
}
|
|
#endif
|