/************************************************************************** Copyright 2001 VA Linux Systems Inc., Fremont, California. Copyright © 2002 by David Dawes All Rights Reserved. Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation on the rights to use, copy, modify, merge, publish, distribute, sub license, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice (including the next paragraph) shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL THE COPYRIGHT HOLDERS AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. **************************************************************************/ /* * Authors: Jeff Hartmann <jhartmann@valinux.com> * Abraham van der Merwe <abraham@2d3d.co.za> * David Dawes <dawes@xfree86.org> * Alan Hourihane <alanh@tungstengraphics.com> */ #ifdef HAVE_CONFIG_H #include "config.h" #endif //#include <memory.h> #include <malloc.h> #include "i915_pciids.h" #include "compiler.h" #include "sna.h" #include "sna_reg.h" #include <pixlib2.h> #include "../pixdriver.h" #include <kos32sys.h> #define to_surface(x) (surface_t*)((x)->handle) typedef struct { int l; int t; int r; int b; } rect_t; static struct sna_fb sna_fb; static int tls_mask; int tls_alloc(void); static inline void *tls_get(int key) { void *val; __asm__ __volatile__( "movl %%fs:(%1), %0" :"=r"(val) :"r"(key)); return val; }; static inline int tls_set(int key, const void *ptr) { if(!(key & 3)) { __asm__ __volatile__( "movl %0, %%fs:(%1)" ::"r"(ptr),"r"(key)); return 0; } else return -1; } int kgem_init_fb(struct kgem *kgem, struct sna_fb *fb); int kgem_update_fb(struct kgem *kgem, struct sna_fb *fb); uint32_t kgem_surface_size(struct kgem *kgem,bool relaxed_fencing, unsigned flags, uint32_t width, uint32_t height, uint32_t bpp, uint32_t tiling, uint32_t *pitch); struct kgem_bo *kgem_bo_from_handle(struct kgem *kgem, int handle, int pitch, int height); void kgem_close_batches(struct kgem *kgem); void sna_bo_destroy(struct kgem *kgem, struct kgem_bo *bo); static bool sna_solid_cache_init(struct sna *sna); struct sna *sna_device; __LOCK_INIT_RECURSIVE(, __sna_lock); static void no_render_reset(struct sna *sna) { (void)sna; } static void no_render_flush(struct sna *sna) { (void)sna; } static void no_render_context_switch(struct kgem *kgem, int new_mode) { if (!kgem->nbatch) return; if (kgem_ring_is_idle(kgem, kgem->ring)) { DBG(("%s: GPU idle, flushing\n", __FUNCTION__)); _kgem_submit(kgem); } (void)new_mode; } static void no_render_retire(struct kgem *kgem) { (void)kgem; } static void no_render_expire(struct kgem *kgem) { (void)kgem; } static void no_render_fini(struct sna *sna) { (void)sna; } const char *no_render_init(struct sna *sna) { struct sna_render *render = &sna->render; memset (render,0, sizeof (*render)); render->prefer_gpu = PREFER_GPU_BLT; render->vertices = render->vertex_data; render->vertex_size = ARRAY_SIZE(render->vertex_data); render->reset = no_render_reset; render->flush = no_render_flush; render->fini = no_render_fini; sna->kgem.context_switch = no_render_context_switch; sna->kgem.retire = no_render_retire; sna->kgem.expire = no_render_expire; sna->kgem.mode = KGEM_RENDER; sna->kgem.ring = KGEM_RENDER; sna_vertex_init(sna); return "generic"; } void sna_vertex_init(struct sna *sna) { // pthread_mutex_init(&sna->render.lock, NULL); // pthread_cond_init(&sna->render.wait, NULL); sna->render.active = 0; } int sna_accel_init(struct sna *sna) { const char *backend; backend = no_render_init(sna); if (sna->info->gen >= 0100) (void)backend; else if (sna->info->gen >= 070) backend = gen7_render_init(sna, backend); else if (sna->info->gen >= 060) backend = gen6_render_init(sna, backend); else if (sna->info->gen >= 050) backend = gen5_render_init(sna, backend); else if (sna->info->gen >= 040) backend = gen4_render_init(sna, backend); else if (sna->info->gen >= 030) backend = gen3_render_init(sna, backend); DBG(("%s(backend=%s, prefer_gpu=%x)\n", __FUNCTION__, backend, sna->render.prefer_gpu)); kgem_reset(&sna->kgem); sna_device = sna; return kgem_init_fb(&sna->kgem, &sna_fb); } #if 0 static bool sna_solid_cache_init(struct sna *sna) { struct sna_solid_cache *cache = &sna->render.solid_cache; DBG(("%s\n", __FUNCTION__)); cache->cache_bo = kgem_create_linear(&sna->kgem, sizeof(cache->color)); if (!cache->cache_bo) return FALSE; /* * Initialise [0] with white since it is very common and filling the * zeroth slot simplifies some of the checks. */ cache->color[0] = 0xffffffff; cache->bo[0] = kgem_create_proxy(cache->cache_bo, 0, sizeof(uint32_t)); cache->bo[0]->pitch = 4; cache->dirty = 1; cache->size = 1; cache->last = 0; return TRUE; } void sna_render_flush_solid(struct sna *sna) { struct sna_solid_cache *cache = &sna->render.solid_cache; DBG(("sna_render_flush_solid(size=%d)\n", cache->size)); assert(cache->dirty); assert(cache->size); kgem_bo_write(&sna->kgem, cache->cache_bo, cache->color, cache->size*sizeof(uint32_t)); cache->dirty = 0; cache->last = 0; } static void sna_render_finish_solid(struct sna *sna, bool force) { struct sna_solid_cache *cache = &sna->render.solid_cache; int i; DBG(("sna_render_finish_solid(force=%d, domain=%d, busy=%d, dirty=%d)\n", force, cache->cache_bo->domain, cache->cache_bo->rq != NULL, cache->dirty)); if (!force && cache->cache_bo->domain != DOMAIN_GPU) return; if (cache->dirty) sna_render_flush_solid(sna); for (i = 0; i < cache->size; i++) { if (cache->bo[i] == NULL) continue; kgem_bo_destroy(&sna->kgem, cache->bo[i]); cache->bo[i] = NULL; } kgem_bo_destroy(&sna->kgem, cache->cache_bo); DBG(("sna_render_finish_solid reset\n")); cache->cache_bo = kgem_create_linear(&sna->kgem, sizeof(cache->color)); cache->bo[0] = kgem_create_proxy(cache->cache_bo, 0, sizeof(uint32_t)); cache->bo[0]->pitch = 4; if (force) cache->size = 1; } struct kgem_bo * sna_render_get_solid(struct sna *sna, uint32_t color) { struct sna_solid_cache *cache = &sna->render.solid_cache; int i; DBG(("%s: %08x\n", __FUNCTION__, color)); // if ((color & 0xffffff) == 0) /* alpha only */ // return kgem_bo_reference(sna->render.alpha_cache.bo[color>>24]); if (color == 0xffffffff) { DBG(("%s(white)\n", __FUNCTION__)); return kgem_bo_reference(cache->bo[0]); } if (cache->color[cache->last] == color) { DBG(("sna_render_get_solid(%d) = %x (last)\n", cache->last, color)); return kgem_bo_reference(cache->bo[cache->last]); } for (i = 1; i < cache->size; i++) { if (cache->color[i] == color) { if (cache->bo[i] == NULL) { DBG(("sna_render_get_solid(%d) = %x (recreate)\n", i, color)); goto create; } else { DBG(("sna_render_get_solid(%d) = %x (old)\n", i, color)); goto done; } } } sna_render_finish_solid(sna, i == ARRAY_SIZE(cache->color)); i = cache->size++; cache->color[i] = color; cache->dirty = 1; DBG(("sna_render_get_solid(%d) = %x (new)\n", i, color)); create: cache->bo[i] = kgem_create_proxy(cache->cache_bo, i*sizeof(uint32_t), sizeof(uint32_t)); cache->bo[i]->pitch = 4; done: cache->last = i; return kgem_bo_reference(cache->bo[i]); } #endif int sna_blit_copy(bitmap_t *src_bitmap, int dst_x, int dst_y, int w, int h, int src_x, int src_y) { struct sna_copy_op copy; struct _Pixmap src, dst; struct kgem_bo *src_bo; char proc_info[1024]; int winx, winy; get_proc_info(proc_info); winx = *(uint32_t*)(proc_info+34); winy = *(uint32_t*)(proc_info+38); memset(&src, 0, sizeof(src)); memset(&dst, 0, sizeof(dst)); src.drawable.bitsPerPixel = 32; src.drawable.width = src_bitmap->width; src.drawable.height = src_bitmap->height; dst.drawable.bitsPerPixel = 32; dst.drawable.width = sna_fb.width; dst.drawable.height = sna_fb.height; memset(©, 0, sizeof(copy)); src_bo = (struct kgem_bo*)src_bitmap->handle; if( sna_device->render.copy(sna_device, GXcopy, &src, src_bo, &dst, sna_fb.fb_bo, ©) ) { copy.blt(sna_device, ©, src_x, src_y, w, h, winx+dst_x, winy+dst_y); copy.done(sna_device, ©); } kgem_submit(&sna_device->kgem); return 0; // __asm__ __volatile__("int3"); }; typedef struct { uint32_t width; uint32_t height; void *data; uint32_t pitch; struct kgem_bo *bo; uint32_t bo_size; uint32_t flags; }surface_t; #define MI_LOAD_REGISTER_IMM (0x22<<23) #define MI_WAIT_FOR_EVENT (0x03<<23) static bool sna_emit_wait_for_scanline_hsw(struct sna *sna, rect_t *crtc, int pipe, int y1, int y2, bool full_height) { uint32_t event; uint32_t *b; if (!sna->kgem.has_secure_batches) return false; b = kgem_get_batch(&sna->kgem); sna->kgem.nbatch += 17; switch (pipe) { default: assert(0); case 0: event = 1 << 0; break; case 1: event = 1 << 8; break; case 2: event = 1 << 14; break; } b[0] = MI_LOAD_REGISTER_IMM | 1; b[1] = 0x44050; /* DERRMR */ b[2] = ~event; b[3] = MI_LOAD_REGISTER_IMM | 1; b[4] = 0xa188; /* FORCEWAKE_MT */ b[5] = 2 << 16 | 2; /* The documentation says that the LOAD_SCAN_LINES command * always comes in pairs. Don't ask me why. */ switch (pipe) { default: assert(0); case 0: event = 0 << 19; break; case 1: event = 1 << 19; break; case 2: event = 4 << 19; break; } b[8] = b[6] = MI_LOAD_SCAN_LINES_INCL | event; b[9] = b[7] = (y1 << 16) | (y2-1); switch (pipe) { default: assert(0); case 0: event = 1 << 0; break; case 1: event = 1 << 8; break; case 2: event = 1 << 14; break; } b[10] = MI_WAIT_FOR_EVENT | event; b[11] = MI_LOAD_REGISTER_IMM | 1; b[12] = 0xa188; /* FORCEWAKE_MT */ b[13] = 2 << 16; b[14] = MI_LOAD_REGISTER_IMM | 1; b[15] = 0x44050; /* DERRMR */ b[16] = ~0; sna->kgem.batch_flags |= I915_EXEC_SECURE; return true; } static bool sna_emit_wait_for_scanline_ivb(struct sna *sna, rect_t *crtc, int pipe, int y1, int y2, bool full_height) { uint32_t *b; uint32_t event; uint32_t forcewake; if (!sna->kgem.has_secure_batches) return false; assert(y1 >= 0); assert(y2 > y1); assert(sna->kgem.mode); /* Always program one less than the desired value */ if (--y1 < 0) y1 = crtc->b; y2--; switch (pipe) { default: assert(0); case 0: event = 1 << (full_height ? 3 : 0); break; case 1: event = 1 << (full_height ? 11 : 8); break; case 2: event = 1 << (full_height ? 21 : 14); break; } if (sna->kgem.gen == 071) forcewake = 0x1300b0; /* FORCEWAKE_VLV */ else forcewake = 0xa188; /* FORCEWAKE_MT */ b = kgem_get_batch(&sna->kgem); /* Both the LRI and WAIT_FOR_EVENT must be in the same cacheline */ if (((sna->kgem.nbatch + 6) >> 4) != (sna->kgem.nbatch + 10) >> 4) { int dw = sna->kgem.nbatch + 6; dw = ALIGN(dw, 16) - dw; while (dw--) *b++ = MI_NOOP; } b[0] = MI_LOAD_REGISTER_IMM | 1; b[1] = 0x44050; /* DERRMR */ b[2] = ~event; b[3] = MI_LOAD_REGISTER_IMM | 1; b[4] = forcewake; b[5] = 2 << 16 | 2; b[6] = MI_LOAD_REGISTER_IMM | 1; b[7] = 0x70068 + 0x1000 * pipe; b[8] = (1 << 31) | (1 << 30) | (y1 << 16) | y2; b[9] = MI_WAIT_FOR_EVENT | event; b[10] = MI_LOAD_REGISTER_IMM | 1; b[11] = forcewake; b[12] = 2 << 16; b[13] = MI_LOAD_REGISTER_IMM | 1; b[14] = 0x44050; /* DERRMR */ b[15] = ~0; sna->kgem.nbatch = b - sna->kgem.batch + 16; sna->kgem.batch_flags |= I915_EXEC_SECURE; return true; } static bool sna_emit_wait_for_scanline_gen6(struct sna *sna, rect_t *crtc, int pipe, int y1, int y2, bool full_height) { uint32_t *b; uint32_t event; // if (!sna->kgem.has_secure_batches) // return false; assert(y1 >= 0); assert(y2 > y1); assert(sna->kgem.mode == KGEM_RENDER); /* Always program one less than the desired value */ if (--y1 < 0) y1 = crtc->b; y2--; /* The scanline granularity is 3 bits */ y1 &= ~7; y2 &= ~7; if (y2 == y1) return false; event = 1 << (3*full_height + pipe*8); b = kgem_get_batch(&sna->kgem); sna->kgem.nbatch += 10; b[0] = MI_LOAD_REGISTER_IMM | 1; b[1] = 0x44050; /* DERRMR */ b[2] = ~event; b[3] = MI_LOAD_REGISTER_IMM | 1; b[4] = 0x4f100; /* magic */ b[5] = (1 << 31) | (1 << 30) | pipe << 29 | (y1 << 16) | y2; b[6] = MI_WAIT_FOR_EVENT | event; b[7] = MI_LOAD_REGISTER_IMM | 1; b[8] = 0x44050; /* DERRMR */ b[9] = ~0; sna->kgem.batch_flags |= I915_EXEC_SECURE; return true; } static bool sna_emit_wait_for_scanline_gen4(struct sna *sna, rect_t *crtc, int pipe, int y1, int y2, bool full_height) { uint32_t event; uint32_t *b; if (pipe == 0) { if (full_height) event = MI_WAIT_FOR_PIPEA_SVBLANK; else event = MI_WAIT_FOR_PIPEA_SCAN_LINE_WINDOW; } else { if (full_height) event = MI_WAIT_FOR_PIPEB_SVBLANK; else event = MI_WAIT_FOR_PIPEB_SCAN_LINE_WINDOW; } b = kgem_get_batch(&sna->kgem); sna->kgem.nbatch += 5; /* The documentation says that the LOAD_SCAN_LINES command * always comes in pairs. Don't ask me why. */ b[2] = b[0] = MI_LOAD_SCAN_LINES_INCL | pipe << 20; b[3] = b[1] = (y1 << 16) | (y2-1); b[4] = MI_WAIT_FOR_EVENT | event; return true; } static bool sna_emit_wait_for_scanline_gen2(struct sna *sna, rect_t *crtc, int pipe, int y1, int y2, bool full_height) { uint32_t *b; /* * Pre-965 doesn't have SVBLANK, so we need a bit * of extra time for the blitter to start up and * do its job for a full height blit */ if (full_height) y2 -= 2; b = kgem_get_batch(&sna->kgem); sna->kgem.nbatch += 5; /* The documentation says that the LOAD_SCAN_LINES command * always comes in pairs. Don't ask me why. */ b[2] = b[0] = MI_LOAD_SCAN_LINES_INCL | pipe << 20; b[3] = b[1] = (y1 << 16) | (y2-1); b[4] = MI_WAIT_FOR_EVENT | 1 << (1 + 4*pipe); return true; } bool sna_wait_for_scanline(struct sna *sna, rect_t *crtc, rect_t *clip) { bool full_height; int y1, y2, pipe; bool ret; // if (sna->flags & SNA_NO_VSYNC) // return false; /* * Make sure we don't wait for a scanline that will * never occur */ y1 = clip->t - crtc->t; if (y1 < 1) y1 = 1; y2 = clip->b - crtc->t; if (y2 > crtc->b - crtc->t) y2 = crtc->b - crtc->t; // DBG(("%s: clipped range = %d, %d\n", __FUNCTION__, y1, y2)); // printf("%s: clipped range = %d, %d\n", __FUNCTION__, y1, y2); if (y2 <= y1 + 4) return false; full_height = y1 == 0 && y2 == crtc->b - crtc->t; pipe = sna_fb.pipe; DBG(("%s: pipe=%d, y1=%d, y2=%d, full_height?=%d\n", __FUNCTION__, pipe, y1, y2, full_height)); if (sna->kgem.gen >= 0100) ret = false; else if (sna->kgem.gen >= 075) ret = sna_emit_wait_for_scanline_hsw(sna, crtc, pipe, y1, y2, full_height); else if (sna->kgem.gen >= 070) ret = sna_emit_wait_for_scanline_ivb(sna, crtc, pipe, y1, y2, full_height); else if (sna->kgem.gen >= 060) ret =sna_emit_wait_for_scanline_gen6(sna, crtc, pipe, y1, y2, full_height); else if (sna->kgem.gen >= 040) ret = sna_emit_wait_for_scanline_gen4(sna, crtc, pipe, y1, y2, full_height); else ret = sna_emit_wait_for_scanline_gen2(sna, crtc, pipe, y1, y2, full_height); return ret; } int intel_get_device_id(struct sna *sna) { struct drm_i915_getparam gp; int devid = 0; memset(&gp, 0, sizeof(gp)); gp.param = I915_PARAM_CHIPSET_ID; gp.value = &devid; if (drmIoctl(sna->scrn, DRM_IOCTL_I915_GETPARAM, &gp)) return 0; return devid; } static const struct intel_device_info intel_generic_info = { .gen = -1, }; static const struct intel_device_info intel_i915_info = { .gen = 030, }; static const struct intel_device_info intel_i945_info = { .gen = 031, }; static const struct intel_device_info intel_g33_info = { .gen = 033, }; static const struct intel_device_info intel_i965_info = { .gen = 040, }; static const struct intel_device_info intel_g4x_info = { .gen = 045, }; static const struct intel_device_info intel_ironlake_info = { .gen = 050, }; static const struct intel_device_info intel_sandybridge_info = { .gen = 060, }; static const struct intel_device_info intel_ivybridge_info = { .gen = 070, }; static const struct intel_device_info intel_valleyview_info = { .gen = 071, }; static const struct intel_device_info intel_haswell_info = { .gen = 075, }; #define INTEL_DEVICE_MATCH(d,i) \ { 0x8086, (d), PCI_MATCH_ANY, PCI_MATCH_ANY, 0x3 << 16, 0xff << 16, (intptr_t)(i) } static const struct pci_id_match intel_device_match[] = { INTEL_I915G_IDS(&intel_i915_info), INTEL_I915GM_IDS(&intel_i915_info), INTEL_I945G_IDS(&intel_i945_info), INTEL_I945GM_IDS(&intel_i945_info), INTEL_G33_IDS(&intel_g33_info), INTEL_PINEVIEW_IDS(&intel_g33_info), INTEL_I965G_IDS(&intel_i965_info), INTEL_I965GM_IDS(&intel_i965_info), INTEL_G45_IDS(&intel_g4x_info), INTEL_GM45_IDS(&intel_g4x_info), INTEL_IRONLAKE_D_IDS(&intel_ironlake_info), INTEL_IRONLAKE_M_IDS(&intel_ironlake_info), INTEL_SNB_D_IDS(&intel_sandybridge_info), INTEL_SNB_M_IDS(&intel_sandybridge_info), INTEL_IVB_D_IDS(&intel_ivybridge_info), INTEL_IVB_M_IDS(&intel_ivybridge_info), INTEL_HSW_D_IDS(&intel_haswell_info), INTEL_HSW_M_IDS(&intel_haswell_info), INTEL_VLV_D_IDS(&intel_valleyview_info), INTEL_VLV_M_IDS(&intel_valleyview_info), INTEL_VGA_DEVICE(PCI_MATCH_ANY, &intel_generic_info), { 0, 0, 0 }, }; const struct pci_id_match *PciDevMatch(uint16_t dev,const struct pci_id_match *list) { while(list->device_id) { if(dev==list->device_id) return list; list++; } return NULL; } const struct intel_device_info * intel_detect_chipset(struct pci_device *pci) { const struct pci_id_match *ent = NULL; ent = PciDevMatch(pci->device_id, intel_device_match); if(ent != NULL) return (const struct intel_device_info*)ent->match_data; else return &intel_generic_info; } int drmIoctl(int fd, unsigned long request, void *arg) { ioctl_t io; io.handle = fd; io.io_code = request; io.input = arg; io.inp_size = 64; io.output = NULL; io.out_size = 0; return call_service(&io); } bool gen6_composite(struct sna *sna, uint8_t op, PixmapPtr src, struct kgem_bo *src_bo, PixmapPtr mask,struct kgem_bo *mask_bo, PixmapPtr dst, struct kgem_bo *dst_bo, int32_t src_x, int32_t src_y, int32_t msk_x, int32_t msk_y, int32_t dst_x, int32_t dst_y, int32_t width, int32_t height, struct sna_composite_op *tmp); //#define MAP(ptr) ((void*)((uintptr_t)(ptr) & ~3)) int sna_bitmap_from_handle(bitmap_t *bitmap, uint32_t handle) { surface_t *sf; struct kgem_bo *bo; sf = malloc(sizeof(*sf)); if(sf == NULL) goto err_1; __lock_acquire_recursive(__sna_lock); bo = kgem_bo_from_handle(&sna_device->kgem, handle, bitmap->pitch, bitmap->height); __lock_release_recursive(__sna_lock); sf->width = bitmap->width; sf->height = bitmap->height; sf->data = NULL; sf->pitch = bo->pitch; sf->bo = bo; sf->bo_size = PAGE_SIZE * bo->size.pages.count; sf->flags = bitmap->flags; bitmap->handle = (uint32_t)sf; return 0; err_2: __lock_release_recursive(__sna_lock); free(sf); err_1: return -1; }; void sna_set_bo_handle(bitmap_t *bitmap, int handle) { surface_t *sf = to_surface(bitmap); struct kgem_bo *bo = sf->bo; bo->handle = handle; } static int sna_create_bitmap(bitmap_t *bitmap) { surface_t *sf; struct kgem_bo *bo; sf = malloc(sizeof(*sf)); if(sf == NULL) goto err_1; __lock_acquire_recursive(__sna_lock); bo = kgem_create_2d(&sna_device->kgem, bitmap->width, bitmap->height, 32,I915_TILING_NONE, CREATE_CPU_MAP); if(bo == NULL) goto err_2; void *map = kgem_bo_map(&sna_device->kgem, bo); if(map == NULL) goto err_3; sf->width = bitmap->width; sf->height = bitmap->height; sf->data = map; sf->pitch = bo->pitch; sf->bo = bo; sf->bo_size = PAGE_SIZE * bo->size.pages.count; sf->flags = bitmap->flags; bitmap->handle = (uint32_t)sf; __lock_release_recursive(__sna_lock); return 0; err_3: kgem_bo_destroy(&sna_device->kgem, bo); err_2: __lock_release_recursive(__sna_lock); free(sf); err_1: return -1; }; static int sna_destroy_bitmap(bitmap_t *bitmap) { surface_t *sf = to_surface(bitmap); __lock_acquire_recursive(__sna_lock); kgem_bo_destroy(&sna_device->kgem, sf->bo); __lock_release_recursive(__sna_lock); free(sf); bitmap->handle = -1; bitmap->data = (void*)-1; bitmap->pitch = -1; return 0; }; static int sna_lock_bitmap(bitmap_t *bitmap) { surface_t *sf = to_surface(bitmap); // printf("%s\n", __FUNCTION__); __lock_acquire_recursive(__sna_lock); kgem_bo_sync__cpu(&sna_device->kgem, sf->bo); __lock_release_recursive(__sna_lock); bitmap->data = sf->data; bitmap->pitch = sf->pitch; return 0; }; static int sna_resize_bitmap(bitmap_t *bitmap) { surface_t *sf = to_surface(bitmap); struct kgem *kgem = &sna_device->kgem; struct kgem_bo *bo = sf->bo; uint32_t size; uint32_t pitch; bitmap->pitch = -1; bitmap->data = (void *) -1; size = kgem_surface_size(kgem,kgem->has_relaxed_fencing, CREATE_CPU_MAP, bitmap->width, bitmap->height, 32, I915_TILING_NONE, &pitch); assert(size && size <= kgem->max_object_size); if(sf->bo_size >= size) { sf->width = bitmap->width; sf->height = bitmap->height; sf->pitch = pitch; bo->pitch = pitch; return 0; } else { __lock_acquire_recursive(__sna_lock); sna_bo_destroy(kgem, bo); sf->bo = NULL; bo = kgem_create_2d(kgem, bitmap->width, bitmap->height, 32, I915_TILING_NONE, CREATE_CPU_MAP); if(bo == NULL) { __lock_release_recursive(__sna_lock); return -1; }; void *map = kgem_bo_map(kgem, bo); if(map == NULL) { sna_bo_destroy(kgem, bo); __lock_release_recursive(__sna_lock); return -1; }; __lock_release_recursive(__sna_lock); sf->width = bitmap->width; sf->height = bitmap->height; sf->data = map; sf->pitch = bo->pitch; sf->bo = bo; sf->bo_size = PAGE_SIZE * bo->size.pages.count; } return 0; }; int sna_create_mask() { struct kgem_bo *bo; // printf("%s width %d height %d\n", __FUNCTION__, sna_fb.width, sna_fb.height); __lock_acquire_recursive(__sna_lock); bo = kgem_create_2d(&sna_device->kgem, sna_fb.width, sna_fb.height, 8,I915_TILING_NONE, CREATE_CPU_MAP); if(unlikely(bo == NULL)) goto err_1; int *map = kgem_bo_map(&sna_device->kgem, bo); if(map == NULL) goto err_2; __lock_release_recursive(__sna_lock); memset(map, 0, bo->pitch * sna_fb.height); tls_set(tls_mask, bo); return 0; err_2: kgem_bo_destroy(&sna_device->kgem, bo); err_1: __lock_release_recursive(__sna_lock); return -1; }; int sna_blit_tex(bitmap_t *bitmap, int scale, int vsync, int dst_x, int dst_y,int w, int h, int src_x, int src_y) { surface_t *sf = to_surface(bitmap); struct drm_i915_mask update; struct sna_composite_op composite; struct _Pixmap src, dst, mask; struct kgem_bo *src_bo, *mask_bo; int winx, winy; char proc_info[1024]; get_proc_info(proc_info); winx = *(uint32_t*)(proc_info+34); winy = *(uint32_t*)(proc_info+38); // winw = *(uint32_t*)(proc_info+42)+1; // winh = *(uint32_t*)(proc_info+46)+1; mask_bo = tls_get(tls_mask); if(unlikely(mask_bo == NULL)) { sna_create_mask(); mask_bo = tls_get(tls_mask); if( mask_bo == NULL) return -1; }; if(kgem_update_fb(&sna_device->kgem, &sna_fb)) { __lock_acquire_recursive(__sna_lock); kgem_bo_destroy(&sna_device->kgem, mask_bo); __lock_release_recursive(__sna_lock); sna_create_mask(); mask_bo = tls_get(tls_mask); if( mask_bo == NULL) return -1; } VG_CLEAR(update); update.handle = mask_bo->handle; update.bo_map = (int)kgem_bo_map__cpu(&sna_device->kgem, mask_bo); drmIoctl(sna_device->kgem.fd, SRV_MASK_UPDATE, &update); mask_bo->pitch = update.bo_pitch; memset(&src, 0, sizeof(src)); memset(&dst, 0, sizeof(dst)); memset(&mask, 0, sizeof(dst)); src.drawable.bitsPerPixel = 32; src.drawable.width = sf->width; src.drawable.height = sf->height; dst.drawable.bitsPerPixel = 32; dst.drawable.width = sna_fb.width; dst.drawable.height = sna_fb.height; mask.drawable.bitsPerPixel = 8; mask.drawable.width = update.width; mask.drawable.height = update.height; memset(&composite, 0, sizeof(composite)); src_bo = sf->bo; __lock_acquire_recursive(__sna_lock); if(vsync) { rect_t crtc, clip; crtc.l = 0; crtc.t = 0; crtc.r = sna_fb.width-1; crtc.b = sna_fb.height-1; clip.l = winx+dst_x; clip.t = winy+dst_y; clip.r = clip.l+w-1; clip.b = clip.t+h-1; kgem_set_mode(&sna_device->kgem, KGEM_RENDER, sna_fb.fb_bo); sna_wait_for_scanline(sna_device, &crtc, &clip); } if( sna_device->render.blit_tex(sna_device, PictOpSrc,scale, &src, src_bo, &mask, mask_bo, &dst, sna_fb.fb_bo, src_x, src_y, dst_x, dst_y, winx+dst_x, winy+dst_y, w, h, &composite) ) { struct sna_composite_rectangles r; r.src.x = src_x; r.src.y = src_y; r.mask.x = dst_x; r.mask.y = dst_y; r.dst.x = winx+dst_x; r.dst.y = winy+dst_y; r.width = w; r.height = h; composite.blt(sna_device, &composite, &r); composite.done(sna_device, &composite); }; kgem_submit(&sna_device->kgem); __lock_release_recursive(__sna_lock); bitmap->data = (void*)-1; bitmap->pitch = -1; return 0; } static void sna_fini() { if( sna_device ) { struct kgem_bo *mask; __lock_acquire_recursive(__sna_lock); mask = tls_get(tls_mask); sna_device->render.fini(sna_device); if(mask) kgem_bo_destroy(&sna_device->kgem, mask); // kgem_close_batches(&sna_device->kgem); kgem_cleanup_cache(&sna_device->kgem); sna_device = NULL; __lock_release_recursive(__sna_lock); }; } uint32_t DrvInit(uint32_t service, struct pix_driver *driver) { ioctl_t io; int caps = 0; static struct pci_device device; struct sna *sna; DBG(("%s\n", __FUNCTION__)); __lock_acquire_recursive(__sna_lock); if(sna_device) goto done; io.handle = service; io.io_code = SRV_GET_PCI_INFO; io.input = &device; io.inp_size = sizeof(device); io.output = NULL; io.out_size = 0; if (call_service(&io)!=0) goto err1; sna = malloc(sizeof(*sna)); if (sna == NULL) goto err1; memset(sna, 0, sizeof(*sna)); sna->cpu_features = sna_cpu_detect(); sna->PciInfo = &device; sna->info = intel_detect_chipset(sna->PciInfo); sna->scrn = service; kgem_init(&sna->kgem, service, sna->PciInfo, sna->info->gen); /* Disable tiling by default */ sna->tiling = 0; /* Default fail-safe value of 75 Hz */ // sna->vblank_interval = 1000 * 1000 * 1000 / 75; sna->flags = 0; sna_accel_init(sna); tls_mask = tls_alloc(); // printf("tls mask %x\n", tls_mask); driver->create_bitmap = sna_create_bitmap; driver->destroy_bitmap = sna_destroy_bitmap; driver->lock_bitmap = sna_lock_bitmap; driver->blit = sna_blit_tex; driver->resize_bitmap = sna_resize_bitmap; driver->fini = sna_fini; done: caps = sna_device->render.caps; err1: __lock_release_recursive(__sna_lock); return caps; }