377 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			377 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			C
		
	
	
	
| /***********************************************************
 | |
| 
 | |
| Copyright 1987, 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.
 | |
| 
 | |
| 
 | |
| Copyright 1987 by Digital Equipment Corporation, Maynard, Massachusetts.
 | |
| 
 | |
|                         All Rights Reserved
 | |
| 
 | |
| Permission to use, copy, modify, and distribute this software and its 
 | |
| documentation for any purpose and without fee is hereby granted, 
 | |
| provided that the above copyright notice appear in all copies and that
 | |
| both that copyright notice and this permission notice appear in 
 | |
| supporting documentation, and that the name of Digital not be
 | |
| used in advertising or publicity pertaining to distribution of the
 | |
| software without specific, written prior permission.  
 | |
| 
 | |
| DIGITAL DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, INCLUDING
 | |
| ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN NO EVENT SHALL
 | |
| DIGITAL BE LIABLE FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR
 | |
| ANY DAMAGES WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS,
 | |
| WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION,
 | |
| ARISING OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS
 | |
| SOFTWARE.
 | |
| 
 | |
| ******************************************************************/
 | |
| /* pixmap management
 | |
|    written by drewry, september 1986
 | |
| 
 | |
|    on a monchrome device, a pixmap is a bitmap.
 | |
| */
 | |
| 
 | |
| #include <string.h>
 | |
| 
 | |
| #ifdef HAVE_DIX_CONFIG_H
 | |
| #include <dix-config.h>
 | |
| #endif
 | |
| 
 | |
| #include <X11/Xmd.h>
 | |
| #include "servermd.h"
 | |
| #include "scrnintstr.h"
 | |
| #include "pixmapstr.h"
 | |
| #include "mi.h"
 | |
| #include "cfb.h"
 | |
| #include "cfbmskbits.h"
 | |
| 
 | |
| PixmapPtr
 | |
| cfbCreatePixmap (pScreen, width, height, depth)
 | |
|     ScreenPtr	pScreen;
 | |
|     int		width;
 | |
|     int		height;
 | |
|     int		depth;
 | |
| {
 | |
|     PixmapPtr pPixmap;
 | |
|     size_t datasize;
 | |
|     size_t paddedWidth;
 | |
| 
 | |
|     paddedWidth = PixmapBytePad(width, depth);
 | |
| 
 | |
|     if (paddedWidth / 4 > 32767 || height > 32767)
 | |
| 	return NullPixmap;
 | |
|     datasize = height * paddedWidth;
 | |
|     pPixmap = AllocatePixmap(pScreen, datasize);
 | |
|     if (!pPixmap)
 | |
| 	return NullPixmap;
 | |
|     pPixmap->drawable.type = DRAWABLE_PIXMAP;
 | |
|     pPixmap->drawable.class = 0;
 | |
|     pPixmap->drawable.pScreen = pScreen;
 | |
|     pPixmap->drawable.depth = depth;
 | |
|     pPixmap->drawable.bitsPerPixel = BitsPerPixel(depth);
 | |
|     pPixmap->drawable.id = 0;
 | |
|     pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER;
 | |
|     pPixmap->drawable.x = 0;
 | |
|     pPixmap->drawable.y = 0;
 | |
|     pPixmap->drawable.width = width;
 | |
|     pPixmap->drawable.height = height;
 | |
|     pPixmap->devKind = paddedWidth;
 | |
|     pPixmap->refcnt = 1;
 | |
| #ifdef PIXPRIV
 | |
|     pPixmap->devPrivate.ptr = datasize ?
 | |
| 		(pointer)((char *)pPixmap + pScreen->totalPixmapSize) : NULL;
 | |
| #else
 | |
|     pPixmap->devPrivate.ptr = (pointer)(pPixmap + 1);
 | |
| #endif
 | |
|     return pPixmap;
 | |
| }
 | |
| 
 | |
| Bool
 | |
| cfbDestroyPixmap(pPixmap)
 | |
|     PixmapPtr pPixmap;
 | |
| {
 | |
|     if(--pPixmap->refcnt)
 | |
| 	return TRUE;
 | |
|     xfree(pPixmap);
 | |
|     return TRUE;
 | |
| }
 | |
| 
 | |
| PixmapPtr
 | |
| cfbCopyPixmap(pSrc)
 | |
|     register PixmapPtr	pSrc;
 | |
| {
 | |
|     register PixmapPtr	pDst;
 | |
|     int		size;
 | |
|     ScreenPtr pScreen;
 | |
| 
 | |
|     size = pSrc->drawable.height * pSrc->devKind;
 | |
|     pScreen = pSrc->drawable.pScreen;
 | |
|     pDst = (*pScreen->CreatePixmap) (pScreen, pSrc->drawable.width, 
 | |
| 				pSrc->drawable.height, pSrc->drawable.depth);
 | |
|     if (!pDst)
 | |
| 	return NullPixmap;
 | |
|     memmove((char *)pDst->devPrivate.ptr, (char *)pSrc->devPrivate.ptr, size);
 | |
|     return pDst;
 | |
| }
 | |
| 
 | |
| 
 | |
| /* replicates a pattern to be a full 32 bits wide.
 | |
|    relies on the fact that each scnaline is longword padded.
 | |
|    doesn't do anything if pixmap is not a factor of 32 wide.
 | |
|    changes width field of pixmap if successful, so that the fast
 | |
| 	cfbXRotatePixmap code gets used if we rotate the pixmap later.
 | |
| 	cfbYRotatePixmap code gets used if we rotate the pixmap later.
 | |
| 
 | |
|    calculate number of times to repeat
 | |
|    for each scanline of pattern
 | |
|       zero out area to be filled with replicate
 | |
|       left shift and or in original as many times as needed
 | |
| */
 | |
| void
 | |
| cfbPadPixmap(pPixmap)
 | |
|     PixmapPtr pPixmap;
 | |
| {
 | |
|     register int width = (pPixmap->drawable.width) * (pPixmap->drawable.bitsPerPixel);
 | |
|     register int h;
 | |
|     register CfbBits mask;
 | |
|     register CfbBits *p;
 | |
|     register CfbBits bits; /* real pattern bits */
 | |
|     register int i;
 | |
|     int rep;                    /* repeat count for pattern */
 | |
|  
 | |
|     if (width >= PGSZ)
 | |
|         return;
 | |
| 
 | |
|     rep = PGSZ/width;
 | |
|     if (rep*width != PGSZ)
 | |
|         return;
 | |
|  
 | |
|     mask = mfbGetendtab(width);
 | |
|  
 | |
|     p = (CfbBits *)(pPixmap->devPrivate.ptr);
 | |
|     for (h=0; h < pPixmap->drawable.height; h++)
 | |
|     {
 | |
|         *p &= mask;
 | |
|         bits = *p;
 | |
|         for(i=1; i<rep; i++)
 | |
|         {
 | |
| #if (BITMAP_BIT_ORDER == MSBFirst) 
 | |
|             bits >>= width;
 | |
| #else
 | |
| 	    bits <<= width;
 | |
| #endif
 | |
|             *p |= bits;
 | |
|         }
 | |
|         p++;
 | |
|     }    
 | |
|     pPixmap->drawable.width = PGSZ/(pPixmap->drawable.bitsPerPixel);
 | |
| }
 | |
| 
 | |
| 
 | |
| #ifdef notdef
 | |
| /*
 | |
|  * cfb debugging routine -- assumes pixmap is 1 byte deep 
 | |
|  */
 | |
| static cfbdumppixmap(pPix)
 | |
|     PixmapPtr	pPix;
 | |
| {
 | |
|     unsigned int *pw;
 | |
|     char *psrc, *pdst;
 | |
|     int	i, j;
 | |
|     char	line[66];
 | |
| 
 | |
|     ErrorF(  "pPixmap: 0x%x\n", pPix);
 | |
|     ErrorF(  "%d wide %d high\n", pPix->drawable.width, pPix->drawable.height);
 | |
|     if (pPix->drawable.width > 64)
 | |
|     {
 | |
| 	ErrorF(  "too wide to see\n");
 | |
| 	return;
 | |
|     }
 | |
| 
 | |
|     pw = (unsigned int *) pPix->devPrivate.ptr;
 | |
|     psrc = (char *) pw;
 | |
| 
 | |
| /*
 | |
|     for ( i=0; i<pPix->drawable.height; ++i )
 | |
| 	ErrorF( "0x%x\n", pw[i] );
 | |
| */
 | |
| 
 | |
|     for ( i = 0; i < pPix->drawable.height; ++i ) {
 | |
| 	pdst = line;
 | |
| 	for(j = 0; j < pPix->drawable.width; j++) {
 | |
| 	    *pdst++ = *psrc++ ? 'X' : ' ' ;
 | |
| 	}
 | |
| 	*pdst++ = '\n';
 | |
| 	*pdst++ = '\0';
 | |
| 	ErrorF( "%s", line);
 | |
|     }
 | |
| }
 | |
| #endif /* notdef */
 | |
| 
 | |
| /* Rotates pixmap pPix by w pixels to the right on the screen. Assumes that
 | |
|  * words are PGSZ bits wide, and that the least significant bit appears on the
 | |
|  * left.
 | |
|  */
 | |
| void
 | |
| cfbXRotatePixmap(pPix, rw)
 | |
|     PixmapPtr	pPix;
 | |
|     register int rw;
 | |
| {
 | |
|     register CfbBits	*pw, *pwFinal;
 | |
|     register CfbBits	t;
 | |
|     int				rot;
 | |
| 
 | |
|     if (pPix == NullPixmap)
 | |
|         return;
 | |
| 
 | |
|     switch (((DrawablePtr) pPix)->bitsPerPixel) {
 | |
| 	case PSZ:
 | |
| 	    break;
 | |
| 	case 1:
 | |
| 	    mfbXRotatePixmap(pPix, rw);
 | |
| 	    return;
 | |
| 	default:
 | |
| 	    ErrorF("cfbXRotatePixmap: unsupported bitsPerPixel %d\n", ((DrawablePtr) pPix)->bitsPerPixel);
 | |
| 	    return;
 | |
|     }
 | |
|     pw = (CfbBits *)pPix->devPrivate.ptr;
 | |
|     modulus (rw, (int) pPix->drawable.width, rot);
 | |
|     if(pPix->drawable.width == PPW)
 | |
|     {
 | |
|         pwFinal = pw + pPix->drawable.height;
 | |
| 	while(pw < pwFinal)
 | |
| 	{
 | |
| 	    t = *pw;
 | |
| 	    *pw++ = SCRRIGHT(t, rot) |
 | |
| 		    (SCRLEFT(t, (PPW-rot)) & cfbendtab[rot]);
 | |
| 	}
 | |
|     }
 | |
|     else
 | |
|     {
 | |
|         ErrorF("cfb internal error: trying to rotate odd-sized pixmap.\n");
 | |
| #ifdef notdef
 | |
| 	register CfbBits *pwTmp;
 | |
| 	int size, tsize;
 | |
| 
 | |
| 	tsize = PixmapBytePad(pPix->drawable.width - rot, pPix->drawable.depth);
 | |
| 	pwTmp = (CfbBits *) ALLOCATE_LOCAL(pPix->drawable.height * tsize);
 | |
| 	if (!pwTmp)
 | |
| 	    return;
 | |
| 	/* divide pw (the pixmap) in two vertically at (w - rot) and swap */
 | |
| 	tsize >>= 2;
 | |
| 	size = pPix->devKind >> SIZE0F(PixelGroup);
 | |
| 	cfbQuickBlt((CfbBits *)pw, (CfbBits *)pwTmp,
 | |
| 		    0, 0, 0, 0,
 | |
| 		    (int)pPix->drawable.width - rot, (int)pPix->drawable.height,
 | |
| 		    size, tsize);
 | |
| 	cfbQuickBlt((CfbBits *)pw, (CfbBits *)pw,
 | |
| 		    (int)pPix->drawable.width - rot, 0, 0, 0,
 | |
| 		    rot, (int)pPix->drawable.height,
 | |
| 		    size, size);
 | |
| 	cfbQuickBlt((CfbBits *)pwTmp, (CfbBits *)pw,
 | |
| 		    0, 0, rot, 0,
 | |
| 		    (int)pPix->drawable.width - rot, (int)pPix->drawable.height,
 | |
| 		    tsize, size);
 | |
| 	DEALLOCATE_LOCAL(pwTmp);
 | |
| #endif
 | |
|     }
 | |
| }
 | |
| 
 | |
| /* Rotates pixmap pPix by h lines.  Assumes that h is always less than
 | |
|    pPix->drawable.height
 | |
|    works on any width.
 | |
|  */
 | |
| void
 | |
| cfbYRotatePixmap(pPix, rh)
 | |
|     register PixmapPtr	pPix;
 | |
|     int	rh;
 | |
| {
 | |
|     int nbyDown;	/* bytes to move down to row 0; also offset of
 | |
| 			   row rh */
 | |
|     int nbyUp;		/* bytes to move up to line rh; also
 | |
| 			   offset of first line moved down to 0 */
 | |
|     char *pbase;
 | |
|     char *ptmp;
 | |
|     int	rot;
 | |
| 
 | |
|     if (pPix == NullPixmap)
 | |
| 	return;
 | |
|     switch (((DrawablePtr) pPix)->bitsPerPixel) {
 | |
| 	case PSZ:
 | |
| 	    break;
 | |
| 	case 1:
 | |
| 	    mfbYRotatePixmap(pPix, rh);
 | |
| 	    return;
 | |
| 	default:
 | |
| 	    ErrorF("cfbYRotatePixmap: unsupported bitsPerPixel %d\n", ((DrawablePtr) pPix)->bitsPerPixel);
 | |
| 	    return;
 | |
|     }
 | |
| 
 | |
|     modulus (rh, (int) pPix->drawable.height, rot);
 | |
|     pbase = (char *)pPix->devPrivate.ptr;
 | |
| 
 | |
|     nbyDown = rot * pPix->devKind;
 | |
|     nbyUp = (pPix->devKind * pPix->drawable.height) - nbyDown;
 | |
|     if(!(ptmp = (char *)ALLOCATE_LOCAL(nbyUp)))
 | |
| 	return;
 | |
| 
 | |
|     memmove(ptmp, pbase, nbyUp);		/* save the low rows */
 | |
|     memmove(pbase, pbase+nbyUp, nbyDown);	/* slide the top rows down */
 | |
|     memmove(pbase+nbyDown, ptmp, nbyUp);	/* move lower rows up to row rot */
 | |
|     DEALLOCATE_LOCAL(ptmp);
 | |
| }
 | |
| 
 | |
| void
 | |
| cfbCopyRotatePixmap(psrcPix, ppdstPix, xrot, yrot)
 | |
|     register PixmapPtr psrcPix, *ppdstPix;
 | |
|     int	xrot, yrot;
 | |
| {
 | |
|     register PixmapPtr pdstPix;
 | |
| 
 | |
|     if ((pdstPix = *ppdstPix) &&
 | |
| 	(pdstPix->devKind == psrcPix->devKind) &&
 | |
| 	(pdstPix->drawable.height == psrcPix->drawable.height))
 | |
|     {
 | |
| 	memmove((char *)pdstPix->devPrivate.ptr,
 | |
| 		(char *)psrcPix->devPrivate.ptr,
 | |
| 	      psrcPix->drawable.height * psrcPix->devKind);
 | |
| 	pdstPix->drawable.width = psrcPix->drawable.width;
 | |
| 	pdstPix->drawable.depth = psrcPix->drawable.depth;
 | |
| 	pdstPix->drawable.bitsPerPixel = psrcPix->drawable.bitsPerPixel;
 | |
| 	pdstPix->drawable.serialNumber = NEXT_SERIAL_NUMBER;
 | |
|     }
 | |
|     else
 | |
|     {
 | |
| 	if (pdstPix)
 | |
| 	    /* FIX XBUG 6168 */
 | |
| 	    (*pdstPix->drawable.pScreen->DestroyPixmap)(pdstPix);
 | |
| 	*ppdstPix = pdstPix = cfbCopyPixmap(psrcPix);
 | |
| 	if (!pdstPix)
 | |
| 	    return;
 | |
|     }
 | |
|     cfbPadPixmap(pdstPix);
 | |
|     if (xrot)
 | |
| 	cfbXRotatePixmap(pdstPix, xrot);
 | |
|     if (yrot)
 | |
| 	cfbYRotatePixmap(pdstPix, yrot);
 | |
| }
 |