commit
74f3388a61
@ -0,0 +1,448 @@
|
||||
/*
|
||||
Copyright 2013 Jay Sorg
|
||||
|
||||
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.
|
||||
|
||||
XVideo extension
|
||||
|
||||
*/
|
||||
|
||||
#include "rdp.h"
|
||||
|
||||
#include <X11/extensions/Xv.h>
|
||||
#include <X11/extensions/Xvproto.h>
|
||||
#include "xvdix.h"
|
||||
|
||||
#include <fourcc.h>
|
||||
|
||||
extern rdpScreenInfoRec g_rdpScreen; /* from rdpmain.c */
|
||||
|
||||
static DevPrivateKey g_XvScreenKey;
|
||||
static char g_xv_adaptor_name[] = "xrdp XVideo adaptor";
|
||||
static char g_xv_encoding_name[] = "XV_IMAGE";
|
||||
|
||||
#define GET_XV_SCREEN(pScreen) \
|
||||
((XvScreenPtr)dixLookupPrivate(&(pScreen)->devPrivates, g_XvScreenKey))
|
||||
|
||||
#define LOG_LEVEL 1
|
||||
#define LLOG(_level, _args) \
|
||||
do { if (_level < LOG_LEVEL) { ErrorF _args ; } } while (0)
|
||||
#define LLOGLN(_level, _args) \
|
||||
do { if (_level < LOG_LEVEL) { ErrorF _args ; ErrorF("\n"); } } while (0)
|
||||
|
||||
#define T_NUM_ENCODINGS 1
|
||||
static XvEncodingRec g_encodings[T_NUM_ENCODINGS];
|
||||
|
||||
#define T_NUM_FORMATS 1
|
||||
static XvFormatRec g_formats[T_NUM_FORMATS];
|
||||
|
||||
#define T_NUM_PORTS 1
|
||||
static XvPortRec g_ports[T_NUM_PORTS];
|
||||
|
||||
#define FOURCC_RV15 0x35315652
|
||||
#define FOURCC_RV16 0x36315652
|
||||
#define FOURCC_RV24 0x34325652
|
||||
#define FOURCC_RV32 0x32335652
|
||||
|
||||
#define T_NUM_IMAGES 8
|
||||
static XvImageRec g_images[T_NUM_IMAGES] =
|
||||
{
|
||||
{
|
||||
FOURCC_RV15,XvRGB,LSBFirst,
|
||||
{'R','V','1','5',0,0,0,0,0,0,0,0,0,0,0,0},
|
||||
16, XvPacked, 1, 15, 0x001f, 0x03e0, 0x7c00, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
{'R','V','B',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
|
||||
},
|
||||
{
|
||||
FOURCC_RV16,XvRGB,LSBFirst,
|
||||
{'R','V','1','6',0,0,0,0,0,0,0,0,0,0,0,0},
|
||||
16, XvPacked, 1, 16, 0x001f, 0x07e0, 0xf800, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
{'R','V','B',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
|
||||
},
|
||||
{
|
||||
FOURCC_RV24,XvRGB,LSBFirst,
|
||||
{'R','V','2','4',0,0,0,0,0,0,0,0,0,0,0,0},
|
||||
32, XvPacked, 1, 24, 0x000000ff, 0x0000ff00, 0x00ff0000, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
{'R','V','B',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
|
||||
},
|
||||
{
|
||||
FOURCC_RV32, XvRGB, LSBFirst,
|
||||
{'R','V','3','2',0,0,0,0,0,0,0,0,0,0,0,0},
|
||||
32, XvPacked, 1, 32, 0x000000ff, 0x0000ff00, 0x00ff0000, 0, 0, 0, 0, 0, 0, 0, 0, 0,
|
||||
{'R','V','B',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
|
||||
},
|
||||
XVIMAGE_YV12,
|
||||
XVIMAGE_YUY2,
|
||||
XVIMAGE_UYVY,
|
||||
XVIMAGE_I420
|
||||
};
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvAllocatePort(unsigned long port, XvPortPtr pPort, XvPortPtr* ppPort)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvAllocatePort:"));
|
||||
*ppPort = pPort;
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvFreePort(XvPortPtr pPort)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvFreePort:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvPutVideo(ClientPtr client, DrawablePtr pDraw, XvPortPtr pPort, GCPtr pGC,
|
||||
INT16 vid_x, INT16 vid_y, CARD16 vid_w, CARD16 vid_h,
|
||||
INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvPutVideo:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvPutStill(ClientPtr client, DrawablePtr pDraw, XvPortPtr pPort, GCPtr pGC,
|
||||
INT16 vid_x, INT16 vid_y, CARD16 vid_w, CARD16 vid_h,
|
||||
INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvPutStill:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvGetVideo(ClientPtr client, DrawablePtr pDraw, XvPortPtr pPort, GCPtr pGC,
|
||||
INT16 vid_x, INT16 vid_y, CARD16 vid_w, CARD16 vid_h,
|
||||
INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvGetVideo:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvGetStill(ClientPtr client, DrawablePtr pDraw, XvPortPtr pPort, GCPtr pGC,
|
||||
INT16 vid_x, INT16 vid_y, CARD16 vid_w, CARD16 vid_h,
|
||||
INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvGetStill:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvStopVideo(ClientPtr client, XvPortPtr pPort, DrawablePtr pDraw)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvStopVideo:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvSetPortAttribute(ClientPtr client, XvPortPtr pPort, Atom attribute,
|
||||
INT32 value)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvxSetPortAttribute:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvGetPortAttribute(ClientPtr client, XvPortPtr pPort, Atom attribute,
|
||||
INT32* p_value)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvGetPortAttribute:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvQueryBestSize(ClientPtr client, XvPortPtr pPort, CARD8 motion,
|
||||
CARD16 vid_w, CARD16 vid_h, CARD16 drw_w, CARD16 drw_h,
|
||||
unsigned int *p_w, unsigned int *p_h)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvQueryBestSize:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvPutImage(ClientPtr client, DrawablePtr pDraw, XvPortPtr pPort, GCPtr pGC,
|
||||
INT16 src_x, INT16 src_y, CARD16 src_w, CARD16 src_h,
|
||||
INT16 drw_x, INT16 drw_y, CARD16 drw_w, CARD16 drw_h,
|
||||
XvImagePtr format, unsigned char* data, Bool sync,
|
||||
CARD16 width, CARD16 height)
|
||||
{
|
||||
LLOGLN(0, ("rdpXvPutImage:"));
|
||||
return Success;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvPrintFormat(int id)
|
||||
{
|
||||
switch (id)
|
||||
{
|
||||
case FOURCC_YV12:
|
||||
/* big buck bunny 480p h264 */
|
||||
/* vob files too */
|
||||
LLOGLN(0, ("FOURCC_YV12"));
|
||||
break;
|
||||
case FOURCC_I420:
|
||||
LLOGLN(0, ("FOURCC_I420"));
|
||||
break;
|
||||
case FOURCC_RV15:
|
||||
LLOGLN(0, ("FOURCC_RV15"));
|
||||
break;
|
||||
case FOURCC_RV16:
|
||||
LLOGLN(0, ("FOURCC_RV16"));
|
||||
break;
|
||||
case FOURCC_YUY2:
|
||||
LLOGLN(0, ("FOURCC_YUY2"));
|
||||
break;
|
||||
case FOURCC_UYVY:
|
||||
LLOGLN(0, ("FOURCC_UYVY"));
|
||||
break;
|
||||
case FOURCC_RV24:
|
||||
LLOGLN(0, ("FOURCC_RV24"));
|
||||
break;
|
||||
default:
|
||||
LLOGLN(0, ("other"));
|
||||
break;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvQueryImageAttributes(ClientPtr client, XvPortPtr pPort, XvImagePtr format,
|
||||
CARD16* width, CARD16* height, int* pitches,
|
||||
int* offsets)
|
||||
{
|
||||
int size;
|
||||
int tmp;
|
||||
|
||||
LLOGLN(0, ("rdpXvQueryImageAttributes:"));
|
||||
|
||||
|
||||
size = 0;
|
||||
/* this is same code as all drivers currently have */
|
||||
if (*width > 2046)
|
||||
{
|
||||
*width = 2046;
|
||||
}
|
||||
if (*height > 2046)
|
||||
{
|
||||
*height = 2046;
|
||||
}
|
||||
/* make w multiple of 4 so that resizing works properly */
|
||||
*width = (*width + 3) & ~3;
|
||||
if (offsets)
|
||||
{
|
||||
offsets[0] = 0;
|
||||
}
|
||||
LLOGLN(0, ("format %x", format->id));
|
||||
rdpXvPrintFormat(format->id);
|
||||
switch (format->id)
|
||||
{
|
||||
case FOURCC_YV12:
|
||||
case FOURCC_I420:
|
||||
/* make h be even */
|
||||
*height = (*height + 1) & ~1;
|
||||
/* make w be multiple of 4 (ie. pad it) */
|
||||
size = (*width + 3) & ~3;
|
||||
/* width of a Y row => width of image */
|
||||
if (pitches != 0)
|
||||
{
|
||||
pitches[0] = size;
|
||||
}
|
||||
/* offset of U plane => w*h */
|
||||
size *= *height;
|
||||
if (offsets != 0)
|
||||
{
|
||||
offsets[1] = size;
|
||||
}
|
||||
/* width of U, V row => width/2 */
|
||||
tmp = ((*width >> 1) +3) & ~3;
|
||||
if (pitches != 0)
|
||||
{
|
||||
pitches[1] = pitches[2] = tmp;
|
||||
}
|
||||
/* offset of V => Y plane + U plane (w*h + w/2*h/2) */
|
||||
tmp *= (*height >> 1);
|
||||
size += tmp;
|
||||
size += tmp;
|
||||
if (offsets != 0)
|
||||
{
|
||||
offsets[2] = size;
|
||||
}
|
||||
size += tmp;
|
||||
break;
|
||||
case FOURCC_RV15:
|
||||
case FOURCC_RV16:
|
||||
case FOURCC_YUY2:
|
||||
case FOURCC_UYVY:
|
||||
size = (*width) * 2;
|
||||
if (pitches)
|
||||
{
|
||||
pitches[0] = size;
|
||||
}
|
||||
size *= *height;
|
||||
break;
|
||||
case FOURCC_RV24:
|
||||
size = (*width) * 3;
|
||||
if (pitches)
|
||||
{
|
||||
pitches[0] = size;
|
||||
}
|
||||
size *= *height;
|
||||
break;
|
||||
default:
|
||||
LLOGLN(0, ("rdpXvQueryImageAttributes: error"));
|
||||
break;
|
||||
}
|
||||
return size;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static Bool
|
||||
rdpXvCloseScreen(int i, ScreenPtr pScreen)
|
||||
{
|
||||
XvScreenPtr pxvs = GET_XV_SCREEN(pScreen);
|
||||
|
||||
LLOGLN(0, ("rdpXvCloseScreen:"));
|
||||
free(pxvs->pAdaptors);
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvQueryAdaptors(ScreenPtr pScreen, XvAdaptorPtr* p_pAdaptors,
|
||||
int* p_nAdaptors)
|
||||
{
|
||||
XvScreenPtr pxvs = GET_XV_SCREEN(pScreen);
|
||||
|
||||
LLOGLN(0, ("rdpXvQueryAdaptors:"));
|
||||
*p_nAdaptors = pxvs->nAdaptors;
|
||||
*p_pAdaptors = pxvs->pAdaptors;
|
||||
return Success;
|
||||
}
|
||||
|
||||
#ifdef MITSHM
|
||||
#endif
|
||||
|
||||
/*****************************************************************************/
|
||||
static int
|
||||
rdpXvInitAdaptors(ScreenPtr pScreen)
|
||||
{
|
||||
XvScreenPtr pxvs = GET_XV_SCREEN(pScreen);
|
||||
XvAdaptorPtr pAdaptor;
|
||||
|
||||
pAdaptor = malloc(sizeof(XvAdaptorRec));
|
||||
memset(pAdaptor, 0, sizeof(XvAdaptorRec));
|
||||
pAdaptor->type = XvInputMask | XvOutputMask | XvImageMask |
|
||||
XvVideoMask | XvStillMask;
|
||||
pAdaptor->pScreen = pScreen;
|
||||
|
||||
pAdaptor->name = g_xv_adaptor_name;
|
||||
|
||||
pAdaptor->nEncodings = T_NUM_ENCODINGS;
|
||||
pAdaptor->pEncodings = g_encodings;
|
||||
|
||||
pAdaptor->nFormats = T_NUM_FORMATS;
|
||||
pAdaptor->pFormats = g_formats;
|
||||
|
||||
pAdaptor->nImages = T_NUM_IMAGES;
|
||||
pAdaptor->pImages = g_images;
|
||||
|
||||
pAdaptor->nPorts = T_NUM_PORTS;
|
||||
pAdaptor->pPorts = g_ports;
|
||||
|
||||
pAdaptor->ddAllocatePort = rdpXvAllocatePort;
|
||||
pAdaptor->ddFreePort = rdpXvFreePort;
|
||||
pAdaptor->ddPutVideo = rdpXvPutVideo;
|
||||
pAdaptor->ddPutStill = rdpXvPutStill;
|
||||
pAdaptor->ddGetVideo = rdpXvGetVideo;
|
||||
pAdaptor->ddGetStill = rdpXvGetStill;
|
||||
pAdaptor->ddStopVideo = rdpXvStopVideo;
|
||||
pAdaptor->ddSetPortAttribute = rdpXvSetPortAttribute;
|
||||
pAdaptor->ddGetPortAttribute = rdpXvGetPortAttribute;
|
||||
pAdaptor->ddQueryBestSize = rdpXvQueryBestSize;
|
||||
pAdaptor->ddPutImage = rdpXvPutImage;
|
||||
pAdaptor->ddQueryImageAttributes = rdpXvQueryImageAttributes;
|
||||
|
||||
pxvs->pAdaptors = pAdaptor;
|
||||
|
||||
memset(g_encodings, 0, sizeof(g_encodings));
|
||||
g_encodings[0].id = FakeClientID(0);
|
||||
g_encodings[0].pScreen = pScreen;
|
||||
g_encodings[0].name = g_xv_encoding_name;
|
||||
g_encodings[0].width = 2046;
|
||||
g_encodings[0].height = 2046;
|
||||
g_encodings[0].rate.numerator = 1;
|
||||
g_encodings[0].rate.denominator = 1;
|
||||
|
||||
memset(g_formats, 0, sizeof(g_formats));
|
||||
g_formats[0].depth = g_rdpScreen.depth;
|
||||
g_formats[0].visual = pScreen->rootVisual;
|
||||
|
||||
memset(g_ports, 0, sizeof(g_ports));
|
||||
g_ports[0].id = FakeClientID(0);
|
||||
g_ports[0].pAdaptor = pAdaptor;
|
||||
g_ports[0].pNotify = 0;
|
||||
g_ports[0].pDraw = 0;
|
||||
g_ports[0].grab.id = 0;
|
||||
g_ports[0].grab.client = 0;
|
||||
g_ports[0].time = currentTime;
|
||||
g_ports[0].devPriv.ptr = 0;
|
||||
|
||||
pAdaptor->base_id = g_ports[0].id;
|
||||
|
||||
AddResource(g_ports[0].id, XvRTPort, g_ports);
|
||||
|
||||
pxvs->nAdaptors = 1;
|
||||
pxvs->pAdaptors = pAdaptor;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*****************************************************************************/
|
||||
/* returns error */
|
||||
int
|
||||
rdpXvInit(ScreenPtr pScreen)
|
||||
{
|
||||
XvScreenPtr pxvs;
|
||||
|
||||
LLOGLN(0, ("rdpXvInit:"));
|
||||
XvScreenInit(pScreen);
|
||||
g_XvScreenKey = XvGetScreenKey();
|
||||
pxvs = GET_XV_SCREEN(pScreen);
|
||||
pxvs->nAdaptors = 0;
|
||||
pxvs->ddCloseScreen = rdpXvCloseScreen;
|
||||
pxvs->ddQueryAdaptors = rdpXvQueryAdaptors;
|
||||
rdpXvInitAdaptors(pScreen);
|
||||
return 0;
|
||||
}
|
Loading…
Reference in new issue