Mercurial > MadButterfly
diff src/graph_engine_openvg.c @ 618:35a67a837a53 openvg
Use reference count to trace life of _ge_openvg_img_t objects
author | Thinker K.F. Li <thinker@branda.to> |
---|---|
date | Thu, 08 Jul 2010 13:51:47 +0800 |
parents | 7d70a811829b |
children | 7020ed3c3e37 |
line wrap: on
line diff
--- a/src/graph_engine_openvg.c Thu Jul 08 11:56:30 2010 +0800 +++ b/src/graph_engine_openvg.c Thu Jul 08 13:51:47 2010 +0800 @@ -26,6 +26,75 @@ (mtx)[8] = 1; \ } while(0) +/*! \brief Convert mb_img_fmt_t to VGImageFormat */ +static VGImageFormat +_mb_ifmt_2_vgifmt(mb_img_fmt_t fmt) { + VGImageFormat vgifmt; + + switch(fmt) { + case MB_IFMT_ARGB32: + vgifmt = VG_sARGB_8888; + break; + + case MB_IFMT_RGB24: + vgifmt = -1; + break; + + case MB_IFMT_A8: + vgifmt = VG_A_8; + break; + + case MB_IFMT_A1: + vgifmt = -1; + break; + + case MB_IFMT_RGB16_565: + vgifmt = VG_sRGB_565; + break; + + default: + return -1; + } + + return vgifmt; +} + +/*! \brief create image object for OpenVG */ +static _ge_openvg_img_t * +_alloc_vgimage(mb_img_fmt_t fmt, int w, int h) { + VGImage vg_img; + VGImageFormat vgifmt; + _ge_openvg_img_t *ge_img; + + vgifmt = _mb_ifmt_2_vgifmt(fmt); + if(vgifmt == -1) + return NULL; + vg_img = vgCreateImage(vgifmt, w, h, + VG_IMAGE_QUALITY_NONANTIALIASED); + if(vg_img == VG_INVALID_HANDLE) + return NULL; + ge_img = O_ALLOC(_ge_openvg_img_t); + if(ge_img == NULL) { + vgDestroyImage(vg_img); + return NULL; + } + ge_img->ref = 1; + ge_img->img = vg_img; + ge_img->asso_pattern = NULL; + ge_img->asso_surface = NULL; + + return ge_img; +} + +/*! \brief Free image object for OpenVG */ +static void +_free_vgimage(_ge_openvg_img_t *ge_img) { + if(--ge_img->ref > 0) + return; + vgDestroyImage(ge_img->img); + free(ge_img); +} + /* * This implementation supports only from image surface. */ @@ -47,6 +116,7 @@ ge_img = surface->asso_img; pattern = O_ALLOC(mbe_pattern_t); pattern->asso_img = ge_img; + ge_img->ref++; /* increase reference count */ pattern->paint = paint; mtx = pattern->mtx; @@ -166,34 +236,25 @@ if(img->fmt != MB_IFMT_ARGB32) return NULL; - /* Create and copy pixels into VGImage */ - vg_img = vgCreateImage(fmt, img->w, img->h, - VG_IMAGE_QUALITY_NONANTIALIASED); - if(vg_img == VG_INVALID_HANDLE) - return NULL; - vgImageSubData(vg_img, img->content, img->stride, fmt, - 0, 0, img->w, img->h); - /* Allocate objects */ - ge_img = O_ALLOC(_ge_openvg_img_t); + ge_img = _alloc_vgimage(MB_IFMT_ARGB32, img->w, img->h); pattern = O_ALLOC(mbe_pattern_t); paint = vgCreatePaint(); if(ge_img == NULL || pattern == NULL || paint == VG_INVALID_HANDLE) goto err; - - - /* Initialize objects */ - ge_img->img = vg_img; - ge_img->asso_pattern = NULL; - ge_img->asso_surface = NULL; - + + /* Create and copy pixels into VGImage */ + vg_img = ge_img->img; + vgImageSubData(vg_img, img->content, img->stride, fmt, + 0, 0, img->w, img->h); + pattern->paint = paint; pattern->asso_img = ge_img; return pattern; err: - if(ge_img) free(ge_img); + if(ge_img) _free_vgimage(ge_img); if(pattern) free(pattern); if(paint != VG_INVALID_HANDLE) vgDestroyPaint(paint); vgDestroyImage(vg_img); @@ -201,6 +262,12 @@ } void +mbe_pattern_destroy(mbe_pattern_t *ptn) { + if(ptn->asso_img) { + } +} + +void mbe_pattern_set_matrix(mbe_pattern_t *ptn, co_aix *mtx) { co_aix rev[6]; @@ -443,6 +510,7 @@ EGLDisplay display; EGLConfig config; EGLint attrib_list[5]; + _ge_openvg_img_t *ge_img; mbe_surface_t *mbe_surface; int r; @@ -450,7 +518,11 @@ r = _openvg_find_config(fmt, w, h, &config); if(r != 0) return NULL; - + + ge_img = _alloc_vgimage(fmt, w, h); + if(ge_img == NULL) + return NULL; + display = _VG_DISPLAY(); attrib_list[0] = EGL_WIDTH; attrib_list[1] = w; @@ -460,23 +532,47 @@ /* Some implementation does not support pbuffer. * We need use some other surface to replace this one. */ - surface = eglCreatePbufferSurface(display, config, attrib_list); - if(surface == EGL_NO_SURFACE) + surface = eglCreatePbufferFromClientBuffer(display, EGL_OPENVG_IMAGE, + (EGLClientBuffer)ge_img->img, + config, attrib_list); + if(surface == EGL_NO_SURFACE) { + _free_vgimage(ge_img); return NULL; + } mbe_surface = O_ALLOC(mbe_surface_t); if(mbe_surface == NULL) { + _free_vgimage(ge_img); eglDestroySurface(display, surface); return NULL; } mbe_surface->surface = surface; mbe_surface->asso_mbe = NULL; + mbe_surface->asso_img = ge_img; mbe_surface->w = w; mbe_surface->h = h; return mbe_surface; } +void +mbe_surface_destroy(mbe_surface_t *surface) { + EGLDisplay display; + + display = _VG_DISPLAY(); + eglDestroySurface(display, surface->surface); + + if(surface->asso_mbe) + surface->asso_mbe->tgt = NULL; + + if(surface->asso_img) { + surface->asso_img->asso_surface = NULL; + _free_vgimage(surface->asso_img); + } + + free(surface); +} + mbe_t * mbe_create(mbe_surface_t *surface) { EGLDisplay display;