/* * Copyright © 2005 Novell, Inc. * * 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, and that the name of * Novell, Inc. not be used in advertising or publicity pertaining to * distribution of the software without specific, written prior permission. * Novell, Inc. makes no representations about the suitability of this * software for any purpose. It is provided "as is" without express or * implied warranty. * * NOVELL, INC. DISCLAIMS ALL WARRANTIES WITH REGARD TO THIS SOFTWARE, * INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS, IN * NO EVENT SHALL NOVELL, INC. 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. * * Authors: David Reveman <davidr@novell.com> * Matthias Hopf <mhopf@suse.de> */ #include "xgl.h" #ifdef XV #include "xvdix.h" #include "gcstruct.h" #include "dixstruct.h" #include <X11/extensions/Xv.h> #include <X11/extensions/Xvproto.h> static DevPrivateKey xglXvScreenKey; static unsigned long portResource = 0; #define XGL_GET_XV_SCREEN(pScreen) ((XvScreenPtr) \ dixLookupPrivate(&(pScreen)->devPrivates, xglXvScreenKey)) #define XGL_XV_SCREEN(pScreen) \ XvScreenPtr pXvScreen = XGL_GET_XV_SCREEN (pScreen) #define XGL_GET_XV_SCREEN_PRIV(pScreen) \ ((xglXvScreenPtr) (GET_XV_SCREEN (pScreen)->devPriv.ptr)) #define XGL_XV_SCREEN_PRIV(pScreen) \ xglXvScreenPtr pXvScreenPriv = XGL_GET_XV_SCREEN_PRIV (pScreen) #define XGL_GET_XV_PORT_PRIV(pPort) \ ((xglXvPortPtr) ((pPort)->devPriv.ptr)) #define XGL_XV_PORT_PRIV(pPort) \ xglXvPortPtr pPortPriv = XGL_GET_XV_PORT_PRIV (pPort) #define XGL_XV_NUM_PORTS 32 #define XGL_XV_IMAGE_MAX_WIDTH 2048 #define XGL_XV_IMAGE_MAX_HEIGHT 2048 static XvImageRec xvImages[] = { { GLITZ_FOURCC_YUY2, XvYUV, BITMAP_BIT_ORDER, { 'Y','U','Y','2', 0x00, 0x00, 0x00, 0x10, 0x80, 0x00, 0x00, 0xAA, 0x00, 0x38, 0x9B, 0x71 }, 16, XvPacked, 1, 0, 0, 0, 0, 8, 8, 8, 1, 2, 2, 1, 1, 1, { 'Y', 'U', 'Y', 'V', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, XvTopToBottom }, { GLITZ_FOURCC_YV12, XvYUV, BITMAP_BIT_ORDER, { 'Y', 'V', '1', '2', 0x00, 0x00, 0x00, 0x10, 0x80, 0x00, 0x00, 0xAA, 0x00, 0x38, 0x9B, 0x71 }, 12, XvPlanar, 3, 0, 0, 0, 0, 8, 8, 8, 1, 2, 2, 1, 2, 2, { 'Y', 'V', 'U', 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, XvTopToBottom }, { GLITZ_FOURCC_RGB, XvRGB, BITMAP_BIT_ORDER, { 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x10, 0x80, 0x00, 0x00, 0xAA, 0x00, 0x38, 0x9B, 0x71 }, 32, XvPacked, 1, 24, 0xff0000, 0xff00, 0xff, 0, 0, 0, 0, 0, 0, 0, 0, 0, { 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0 }, XvTopToBottom } }; static struct _xglXvFormat { CARD32 format; glitz_fourcc_t fourcc; xglPixelFormatRec pixel; } xglXvFormat[XGL_XV_FORMAT_NUM] = { { PICT_yuy2, GLITZ_FOURCC_YUY2, { 16, 6, { 16, 0x00000000, 0x00000000, 0x00000000, 0x00000000, } } }, { PICT_yv12, GLITZ_FOURCC_YV12, { 12, 4, { 12, 0x00000000, 0x00000000, 0x00000000, 0x00000000, } } }, { PICT_x8r8g8b8, GLITZ_FOURCC_RGB, { 24, 8, { 32, 0x00000000, 0x00ff0000, 0x0000ff00, 0x000000ff, } } } }; static int xglXvQueryAdaptors (ScreenPtr pScreen, XvAdaptorPtr *pAdaptors, int *nAdaptors) { XGL_XV_SCREEN (pScreen); *nAdaptors = pXvScreen->nAdaptors; *pAdaptors = pXvScreen->pAdaptors; return Success; } static int xglXvAllocatePort (unsigned long port, XvPortPtr pPort, XvPortPtr *ppPort) { *ppPort = pPort; return Success; } static int xglXvFreePort (XvPortPtr pPort) { XGL_XV_PORT_PRIV (pPort); if (pPortPriv->pDst) { FreePicture ((pointer) pPortPriv->pDst, 0); pPortPriv->pDst = (PicturePtr) 0; } if (pPortPriv->pSrc) { FreePicture ((pointer) pPortPriv->pSrc, 0); pPortPriv->pSrc = (PicturePtr) 0; } if (pPortPriv->pPixmap) { ScreenPtr pScreen; pScreen = pPortPriv->pPixmap->drawable.pScreen; (*pScreen->DestroyPixmap) (pPortPriv->pPixmap); pPortPriv->pPixmap = (PixmapPtr) 0; } return Success; } static int xglXvQueryBestSize (ClientPtr client, XvPortPtr pPort, CARD8 motion, CARD16 srcWidth, CARD16 srcHeight, CARD16 dstWidth, CARD16 dstHeight, unsigned int *pWidth, unsigned int *pHeight) { *pWidth = dstWidth; *pHeight = dstHeight; return Success; } static int xglXvStopVideo (ClientPtr client, XvPortPtr pPort, DrawablePtr pDrawable) { xglXvFreePort (pPort); return Success; } static int xglXvPutImage (ClientPtr client, DrawablePtr pDrawable, XvPortPtr pPort, GCPtr pGC, INT16 srcX, INT16 srcY, CARD16 srcWidth, CARD16 srcHeight, INT16 dstX, INT16 dstY, CARD16 dstWidth, CARD16 dstHeight, XvImagePtr pImage, unsigned char *data, Bool sync, CARD16 width, CARD16 height) { ScreenPtr pScreen = pDrawable->pScreen; PictTransform transform; int depth, bpp; CARD32 format; XGL_SCREEN_PRIV (pScreen); XGL_XV_PORT_PRIV (pPort); XGL_DRAWABLE_PIXMAP (pDrawable); XGL_PIXMAP_PRIV (pPixmap); switch (pImage->id) { case GLITZ_FOURCC_YUY2: bpp = depth = 16; format = PICT_yuy2; break; case GLITZ_FOURCC_YV12: depth = bpp = 12; format = PICT_yv12; break; case GLITZ_FOURCC_RGB: depth = 24; bpp = 32; format = PICT_x8r8g8b8; break; default: return BadImplementation; } pPort->pDraw = pDrawable; if (!pPortPriv->pPixmap) { pPortPriv->pPixmap = (*pScreen->CreatePixmap) (pScreen, 0, 0, depth, 0); if (!pPortPriv->pPixmap) return BadAlloc; } (*pScreen->ModifyPixmapHeader) (pPortPriv->pPixmap, srcWidth, srcHeight, depth, bpp, -1, (pointer) data); XGL_GET_PIXMAP_PRIV (pPortPriv->pPixmap)->stride = -srcWidth; pPortPriv->pPixmap->drawable.serialNumber = NEXT_SERIAL_NUMBER; if (!pPortPriv->pSrc || pPortPriv->pSrc->format != format) { PictFormatPtr pFormat; int error; static XID value = RepeatPad; pFormat = PictureMatchFormat (pScreen, depth, format); if (!pFormat) return BadImplementation; if (pPortPriv->pSrc) FreePicture ((pointer) pPortPriv->pSrc, 0); pPortPriv->pSrc = CreatePicture (0, &pPortPriv->pPixmap->drawable, pFormat, CPRepeat, &value, serverClient, &error); if (!pPortPriv->pSrc) { xglXvFreePort (pPort); return error; } SetPictureFilter (pPortPriv->pSrc, FilterBilinear, strlen (FilterBilinear), 0, 0); } if (!pPortPriv->pDst || pPortPriv->pDst->pDrawable != pDrawable) { PictFormatPtr pFormat = 0; int i, error; for (i = 0; i < pScreen->numVisuals; i++) { if (pScreen->visuals[i].nplanes == pDrawable->depth) { pFormat = PictureMatchVisual (pScreen, pDrawable->depth, &pScreen->visuals[i]); break; } } if (!pFormat) return BadImplementation; if (pPortPriv->pDst) FreePicture ((pointer) pPortPriv->pDst, 0); pPortPriv->pDst = CreatePicture (0, pDrawable, pFormat, 0, 0, serverClient, &error); if (!pPortPriv->pDst) { xglXvFreePort (pPort); return error; } } transform.matrix[0][0] = ((srcWidth << 16) + (dstWidth >> 1)) / dstWidth; transform.matrix[0][1] = 0; transform.matrix[0][2] = 0; /* flip Y */ transform.matrix[1][0] = 0; transform.matrix[1][1] = -((srcHeight << 16) + (dstHeight >> 1)) / dstHeight; transform.matrix[1][2] = (srcHeight << 16); transform.matrix[2][0] = 0; transform.matrix[2][1] = 0; transform.matrix[2][2] = 1 << 16; SetPictureTransform (pPortPriv->pSrc, &transform); if (pPixmap != pScreenPriv->pScreenPixmap && !pPixmapPriv->target) xglEnablePixmapAccel (pPixmap, &pScreenPriv->accel.xv); CompositePicture (PictOpSrc, pPortPriv->pSrc, (PicturePtr) 0, pPortPriv->pDst, srcX, srcY, 0, 0, dstX, dstY, dstWidth, dstHeight); return Success; } static int xglXvQueryImageAttributes (ClientPtr client, XvPortPtr pPort, XvImagePtr pImage, CARD16 *width, CARD16 *height, int *pitches, int *offsets) { if (*width > XGL_XV_IMAGE_MAX_WIDTH) *width = XGL_XV_IMAGE_MAX_WIDTH; if (*height > XGL_XV_IMAGE_MAX_HEIGHT) *height = XGL_XV_IMAGE_MAX_HEIGHT; *width = (*width + 7) & ~7; switch (pImage->id) { case GLITZ_FOURCC_YUY2: if (offsets) offsets[0] = 0; if (pitches) pitches[0] = *width * 2; return *width * *height * 2; case GLITZ_FOURCC_YV12: *height = (*height + 1) & ~1; if (offsets) { offsets[0] = 0; offsets[1] = *width * *height; offsets[2] = *width * *height + (*width >> 1) * (*height >> 1); } if (pitches) { pitches[0] = *width; pitches[1] = pitches[2] = *width >> 1; } return *width * *height + (*width >> 1) * *height; case GLITZ_FOURCC_RGB: if (offsets) offsets[0] = 0; if (pitches) pitches[0] = *width * 4; return *width * *height * 4; default: return 0; } } static void xglXvFreeAdaptor (XvAdaptorPtr pAdaptor) { xfree (pAdaptor->pEncodings); xfree (pAdaptor->pFormats); if (pAdaptor->pPorts) xfree (pAdaptor->pPorts); } static Bool xglXvInitAdaptors (ScreenPtr pScreen) { XvAdaptorPtr pAdaptor; xglXvPortPtr pPortPriv; XvPortPtr pPort; XvFormatPtr pFormat; XvEncodingPtr pEncoding; int i; XGL_XV_SCREEN (pScreen); pXvScreen->nAdaptors = 0; pXvScreen->pAdaptors = NULL; pAdaptor = xcalloc (1, sizeof (XvAdaptorRec)); if (!pAdaptor) return FALSE; pAdaptor->type = XvInputMask | XvImageMask; pAdaptor->pScreen = pScreen; pAdaptor->ddAllocatePort = xglXvAllocatePort; pAdaptor->ddFreePort = xglXvFreePort; pAdaptor->ddStopVideo = xglXvStopVideo; pAdaptor->ddPutImage = xglXvPutImage; pAdaptor->ddQueryBestSize = xglXvQueryBestSize; pAdaptor->ddQueryImageAttributes = xglXvQueryImageAttributes; pAdaptor->name = "Xgl Generic Texture Video"; pEncoding = xcalloc (1, sizeof (XvEncodingRec)); if (!pEncoding) return FALSE; pEncoding->id = 0; pEncoding->pScreen = pScreen; pEncoding->name = "XV_IMAGE"; pEncoding->width = XGL_XV_IMAGE_MAX_WIDTH; pEncoding->height = XGL_XV_IMAGE_MAX_HEIGHT; pEncoding->rate.numerator = 1; pEncoding->rate.denominator = 1; pAdaptor->nEncodings = 1; pAdaptor->pEncodings = pEncoding; pAdaptor->nImages = sizeof (xvImages) / sizeof (XvImageRec); pAdaptor->pImages = xvImages; /* TODO: Currently no attributes */ pAdaptor->nAttributes = 0; pAdaptor->pAttributes = 0; pFormat = xcalloc (pScreen->numVisuals, sizeof (XvFormatRec)); if (!pFormat) return FALSE; for (i = 0; i < pScreen->numVisuals; i++) { pFormat[i].depth = pScreen->visuals[i].nplanes; pFormat[i].visual = pScreen->visuals[i].vid; } /* All visuals allowed */ pAdaptor->nFormats = pScreen->numVisuals; pAdaptor->pFormats = pFormat; pPort = xcalloc (XGL_XV_NUM_PORTS, sizeof (XvPortRec) + sizeof (xglXvPortRec)); pPortPriv = (xglXvPortPtr) (pPort + XGL_XV_NUM_PORTS); if (!pPort) return FALSE; for (i = 0; i < XGL_XV_NUM_PORTS; i++) { pPort[i].id = FakeClientID (0); if (!pPort[i].id) return FALSE; if (!AddResource (pPort[i].id, portResource, &pPort[i])) return FALSE; pPort[i].pAdaptor = pAdaptor; pPort[i].pNotify = (XvPortNotifyPtr) 0; pPort[i].pDraw = (DrawablePtr) 0; pPort[i].client = (ClientPtr) 0; pPort[i].grab.client = (ClientPtr) 0; pPort[i].time = currentTime; pPort[i].devPriv.ptr = pPortPriv + i; } pAdaptor->nPorts = XGL_XV_NUM_PORTS; pAdaptor->pPorts = pPort; pAdaptor->base_id = pPort->id; pXvScreen->pAdaptors = pAdaptor; pXvScreen->nAdaptors = 1; return TRUE; } static Bool xglXvCloseScreen (int i, ScreenPtr pScreen) { int j; XGL_XV_SCREEN (pScreen); for (j = 0; j < pXvScreen->nAdaptors; j++) xglXvFreeAdaptor (&pXvScreen->pAdaptors[j]); if (pXvScreen->pAdaptors) xfree (pXvScreen->pAdaptors); return TRUE; } Bool xglXvScreenInit (ScreenPtr pScreen) { XvScreenPtr pXvScreen; xglVisualPtr v; int i, status, vid = 0; XGL_SCREEN_PRIV (pScreen); status = XvScreenInit (pScreen); if (status != Success) return FALSE; xglXvScreenKey = XvGetScreenKey (); portResource = XvGetRTPort (); pXvScreen = XGL_GET_XV_SCREEN (pScreen); /* Anyone initializing the Xv layer must provide these two. The Xv di layer calls them without even checking if they exist! */ pXvScreen->ddCloseScreen = xglXvCloseScreen; pXvScreen->ddQueryAdaptors = xglXvQueryAdaptors; pXvScreen->devPriv.ptr = (pointer) 0; if (!xglXvInitAdaptors (pScreen)) return FALSE; for (v = pScreenPriv->pVisual; v; v = v->next) { if (v->vid > vid) vid = v->vid; } memset (pScreenPriv->pXvVisual, 0, sizeof (pScreenPriv->pXvVisual)); for (i = 0; i < XGL_XV_FORMAT_NUM; i++) { glitz_format_t templ; templ.color.fourcc = xglXvFormat[i].fourcc; pScreenPriv->pXvVisual[i].vid = ++vid; pScreenPriv->pXvVisual[i].pPixel = &xglXvFormat[i].pixel; pScreenPriv->pXvVisual[i].format.surface = glitz_find_format (pScreenPriv->drawable, GLITZ_FORMAT_FOURCC_MASK, &templ, 0); } return TRUE; } #endif