271 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			271 lines
		
	
	
		
			7.4 KiB
		
	
	
	
		
			C
		
	
	
	
| /*
 | |
| 
 | |
| Copyright 1993, 1998  The Open Group
 | |
| 
 | |
| Permission to use, copy, modify, distribute, and sell this software and its
 | |
| documentation for any purpose is hereby granted without fee, provided that
 | |
| the above copyright notice appear in all copies and that both that
 | |
| copyright notice and this permission notice appear in supporting
 | |
| documentation.
 | |
| 
 | |
| The above copyright notice and this permission notice 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 NONINFRINGEMENT.
 | |
| IN NO EVENT SHALL THE OPEN GROUP 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.
 | |
| 
 | |
| Except as contained in this notice, the name of The Open Group shall
 | |
| not be used in advertising or otherwise to promote the sale, use or
 | |
| other dealings in this Software without prior written authorization
 | |
| from The Open Group.
 | |
| 
 | |
| */
 | |
| 
 | |
| #ifdef HAVE_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include <X11/X.h>
 | |
| #include "scrnintstr.h"
 | |
| #include "misc.h"
 | |
| #include "os.h"
 | |
| #include "windowstr.h"
 | |
| #include "resource.h"
 | |
| #include "dixstruct.h"
 | |
| #include "gcstruct.h"
 | |
| #include "servermd.h"
 | |
| #include "site.h"
 | |
| 
 | |
| /*
 | |
|  *  Scratch pixmap management and device independent pixmap allocation
 | |
|  *  function.
 | |
|  */
 | |
| 
 | |
| /* callable by ddx */
 | |
| PixmapPtr
 | |
| GetScratchPixmapHeader(ScreenPtr pScreen, int width, int height, int depth,
 | |
|                        int bitsPerPixel, int devKind, pointer pPixData)
 | |
| {
 | |
|     PixmapPtr pPixmap = pScreen->pScratchPixmap;
 | |
| 
 | |
|     if (pPixmap)
 | |
|         pScreen->pScratchPixmap = NULL;
 | |
|     else
 | |
|         /* width and height of 0 means don't allocate any pixmap data */
 | |
|         pPixmap = (*pScreen->CreatePixmap) (pScreen, 0, 0, depth, 0);
 | |
| 
 | |
|     if (pPixmap) {
 | |
|         if ((*pScreen->ModifyPixmapHeader) (pPixmap, width, height, depth,
 | |
|                                             bitsPerPixel, devKind, pPixData))
 | |
|             return pPixmap;
 | |
|         (*pScreen->DestroyPixmap) (pPixmap);
 | |
|     }
 | |
|     return NullPixmap;
 | |
| }
 | |
| 
 | |
| /* callable by ddx */
 | |
| void
 | |
| FreeScratchPixmapHeader(PixmapPtr pPixmap)
 | |
| {
 | |
|     if (pPixmap) {
 | |
|         ScreenPtr pScreen = pPixmap->drawable.pScreen;
 | |
| 
 | |
|         pPixmap->devPrivate.ptr = NULL; /* lest ddx chases bad ptr */
 | |
|         if (pScreen->pScratchPixmap)
 | |
|             (*pScreen->DestroyPixmap) (pPixmap);
 | |
|         else
 | |
|             pScreen->pScratchPixmap = pPixmap;
 | |
|     }
 | |
| }
 | |
| 
 | |
| Bool
 | |
| CreateScratchPixmapsForScreen(ScreenPtr pScreen)
 | |
| {
 | |
|     unsigned int pixmap_size;
 | |
| 
 | |
|     pixmap_size = sizeof(PixmapRec) + dixScreenSpecificPrivatesSize(pScreen, PRIVATE_PIXMAP);
 | |
|     pScreen->totalPixmapSize =
 | |
|         BitmapBytePad(pixmap_size * 8);
 | |
| 
 | |
|     /* let it be created on first use */
 | |
|     pScreen->pScratchPixmap = NULL;
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| void
 | |
| FreeScratchPixmapsForScreen(ScreenPtr pScreen)
 | |
| {
 | |
|     FreeScratchPixmapHeader(pScreen->pScratchPixmap);
 | |
| }
 | |
| 
 | |
| /* callable by ddx */
 | |
| PixmapPtr
 | |
| AllocatePixmap(ScreenPtr pScreen, int pixDataSize)
 | |
| {
 | |
|     PixmapPtr pPixmap;
 | |
| 
 | |
|     assert(pScreen->totalPixmapSize > 0);
 | |
| 
 | |
|     if (pScreen->totalPixmapSize > ((size_t) - 1) - pixDataSize)
 | |
|         return NullPixmap;
 | |
| 
 | |
|     pPixmap = malloc(pScreen->totalPixmapSize + pixDataSize);
 | |
|     if (!pPixmap)
 | |
|         return NullPixmap;
 | |
| 
 | |
|     dixInitScreenPrivates(pScreen, pPixmap, pPixmap + 1, PRIVATE_PIXMAP);
 | |
|     return pPixmap;
 | |
| }
 | |
| 
 | |
| /* callable by ddx */
 | |
| void
 | |
| FreePixmap(PixmapPtr pPixmap)
 | |
| {
 | |
|     dixFiniPrivates(pPixmap, PRIVATE_PIXMAP);
 | |
|     free(pPixmap);
 | |
| }
 | |
| 
 | |
| PixmapPtr PixmapShareToSlave(PixmapPtr pixmap, ScreenPtr slave)
 | |
| {
 | |
|     PixmapPtr spix;
 | |
|     int ret;
 | |
|     void *handle;
 | |
|     ScreenPtr master = pixmap->drawable.pScreen;
 | |
|     int depth = pixmap->drawable.depth;
 | |
| 
 | |
|     ret = master->SharePixmapBacking(pixmap, slave, &handle);
 | |
|     if (ret == FALSE)
 | |
|         return NULL;
 | |
| 
 | |
|     spix = slave->CreatePixmap(slave, 0, 0, depth,
 | |
|                                CREATE_PIXMAP_USAGE_SHARED);
 | |
|     slave->ModifyPixmapHeader(spix, pixmap->drawable.width,
 | |
|                               pixmap->drawable.height, depth, 0,
 | |
|                               pixmap->devKind, NULL);
 | |
| 
 | |
|     /* have the slave pixmap take a reference on the master pixmap
 | |
|        later we destroy them both at the same time */
 | |
|     pixmap->refcnt++;
 | |
| 
 | |
|     spix->master_pixmap = pixmap;
 | |
| 
 | |
|     ret = slave->SetSharedPixmapBacking(spix, handle);
 | |
|     if (ret == FALSE) {
 | |
|         slave->DestroyPixmap(spix);
 | |
|         return NULL;
 | |
|     }
 | |
| 
 | |
|     return spix;
 | |
| }
 | |
| 
 | |
| Bool
 | |
| PixmapStartDirtyTracking(PixmapPtr src,
 | |
|                          PixmapPtr slave_dst,
 | |
|                          int x, int y)
 | |
| {
 | |
|     ScreenPtr screen = src->drawable.pScreen;
 | |
|     PixmapDirtyUpdatePtr dirty_update;
 | |
| 
 | |
|     dirty_update = calloc(1, sizeof(PixmapDirtyUpdateRec));
 | |
|     if (!dirty_update)
 | |
|         return FALSE;
 | |
| 
 | |
|     dirty_update->src = src;
 | |
|     dirty_update->slave_dst = slave_dst;
 | |
|     dirty_update->x = x;
 | |
|     dirty_update->y = y;
 | |
| 
 | |
|     dirty_update->damage = DamageCreate(NULL, NULL,
 | |
|                                         DamageReportNone,
 | |
|                                         TRUE, src->drawable.pScreen,
 | |
|                                         src->drawable.pScreen);
 | |
|     if (!dirty_update->damage) {
 | |
|         free(dirty_update);
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     DamageRegister(&src->drawable, dirty_update->damage);
 | |
|     xorg_list_add(&dirty_update->ent, &screen->pixmap_dirty_list);
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| Bool
 | |
| PixmapStopDirtyTracking(PixmapPtr src, PixmapPtr slave_dst)
 | |
| {
 | |
|     ScreenPtr screen = src->drawable.pScreen;
 | |
|     PixmapDirtyUpdatePtr ent, safe;
 | |
| 
 | |
|     xorg_list_for_each_entry_safe(ent, safe, &screen->pixmap_dirty_list, ent) {
 | |
|         if (ent->src == src && ent->slave_dst == slave_dst) {
 | |
|             DamageUnregister(&src->drawable, ent->damage);
 | |
|             DamageDestroy(ent->damage);
 | |
|             xorg_list_del(&ent->ent);
 | |
|             free(ent);
 | |
|         }
 | |
|     }
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * this function can possibly be improved and optimised, by clipping
 | |
|  * instead of iterating
 | |
|  */
 | |
| Bool PixmapSyncDirtyHelper(PixmapDirtyUpdatePtr dirty, RegionPtr dirty_region)
 | |
| {
 | |
|     ScreenPtr pScreen = dirty->src->drawable.pScreen;
 | |
|     int n;
 | |
|     BoxPtr b;
 | |
|     RegionPtr region = DamageRegion(dirty->damage);
 | |
|     GCPtr pGC;
 | |
|     PixmapPtr dst;
 | |
|     SourceValidateProcPtr SourceValidate;
 | |
| 
 | |
|     /*
 | |
|      * SourceValidate is used by the software cursor code
 | |
|      * to pull the cursor off of the screen when reading
 | |
|      * bits from the frame buffer. Bypassing this function
 | |
|      * leaves the software cursor in place
 | |
|      */
 | |
|     SourceValidate = pScreen->SourceValidate;
 | |
|     pScreen->SourceValidate = NULL;
 | |
| 
 | |
|     RegionTranslate(dirty_region, dirty->x, dirty->y);
 | |
|     RegionIntersect(dirty_region, dirty_region, region);
 | |
| 
 | |
|     if (RegionNil(dirty_region)) {
 | |
|         RegionUninit(dirty_region);
 | |
|         return FALSE;
 | |
|     }
 | |
| 
 | |
|     dst = dirty->slave_dst->master_pixmap;
 | |
| 
 | |
|     RegionTranslate(dirty_region, -dirty->x, -dirty->y);
 | |
|     n = RegionNumRects(dirty_region);
 | |
|     b = RegionRects(dirty_region);
 | |
| 
 | |
|     pGC = GetScratchGC(dirty->src->drawable.depth, pScreen);
 | |
|     ValidateGC(&dst->drawable, pGC);
 | |
| 
 | |
|     while (n--) {
 | |
|         BoxRec dst_box;
 | |
|         int w, h;
 | |
| 
 | |
|         dst_box = *b;
 | |
|         w = dst_box.x2 - dst_box.x1;
 | |
|         h = dst_box.y2 - dst_box.y1;
 | |
| 
 | |
|         pGC->ops->CopyArea(&dirty->src->drawable, &dst->drawable, pGC,
 | |
|                            dirty->x + dst_box.x1, dirty->y + dst_box.y1, w, h, dst_box.x1, dst_box.y1);
 | |
|         b++;
 | |
|     }
 | |
|     FreeScratchGC(pGC);
 | |
| 
 | |
|     pScreen->SourceValidate = SourceValidate;
 | |
|     return TRUE;
 | |
| }
 |