2014-09-21 04:21:37 +08:00
|
|
|
/*
|
|
|
|
Copyright 2014 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
|
|
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
/* this should be before all X11 .h files */
|
|
|
|
#include <xorg-server.h>
|
|
|
|
#include <xorgVersion.h>
|
|
|
|
|
|
|
|
/* all driver need this */
|
|
|
|
#include <xf86.h>
|
|
|
|
#include <xf86_OSproc.h>
|
|
|
|
|
|
|
|
#include <xf86xv.h>
|
|
|
|
#include <X11/extensions/Xv.h>
|
|
|
|
#include <fourcc.h>
|
|
|
|
|
|
|
|
#include "rdp.h"
|
|
|
|
#include "rdpMisc.h"
|
|
|
|
#include "rdpReg.h"
|
|
|
|
#include "rdpClientCon.h"
|
|
|
|
|
|
|
|
#define LOG_LEVEL 1
|
|
|
|
#define LLOGLN(_level, _args) \
|
|
|
|
do { if (_level < LOG_LEVEL) { ErrorF _args ; ErrorF("\n"); } } while (0)
|
|
|
|
|
|
|
|
#define T_NUM_ENCODINGS 1
|
|
|
|
static XF86VideoEncodingRec g_xrdpVidEncodings[T_NUM_ENCODINGS] =
|
|
|
|
{ { 0, "XV_IMAGE", 2046, 2046, { 1, 1 } } };
|
|
|
|
|
|
|
|
#define T_NUM_FORMATS 1
|
|
|
|
static XF86VideoFormatRec g_xrdpVidFormats[T_NUM_FORMATS] =
|
|
|
|
{ { 0, TrueColor } };
|
|
|
|
|
|
|
|
/* XVIMAGE_YV12 FOURCC_YV12 0x32315659 */
|
|
|
|
/* XVIMAGE_YUY2 FOURCC_YUY2 0x32595559 */
|
|
|
|
/* XVIMAGE_UYVY FOURCC_UYVY 0x59565955 */
|
|
|
|
/* XVIMAGE_I420 FOURCC_I420 0x30323449 */
|
|
|
|
|
|
|
|
static XF86ImageRec g_xrdpVidImages[] =
|
2014-09-21 07:10:20 +08:00
|
|
|
{ XVIMAGE_YV12, XVIMAGE_YUY2, XVIMAGE_UYVY, XVIMAGE_I420 };
|
2014-09-21 04:21:37 +08:00
|
|
|
|
|
|
|
#define T_MAX_PORTS 1
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidPutVideo(ScrnInfoPtr pScrn, short vid_x, short vid_y,
|
|
|
|
short drw_x, short drw_y, short vid_w, short vid_h,
|
|
|
|
short drw_w, short drw_h, RegionPtr clipBoxes,
|
|
|
|
pointer data, DrawablePtr pDraw)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidPutVideo:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidPutStill(ScrnInfoPtr pScrn, short vid_x, short vid_y,
|
|
|
|
short drw_x, short drw_y, short vid_w, short vid_h,
|
|
|
|
short drw_w, short drw_h, RegionPtr clipBoxes,
|
|
|
|
pointer data, DrawablePtr pDraw)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidPutStill:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidGetVideo(ScrnInfoPtr pScrn, short vid_x, short vid_y,
|
|
|
|
short drw_x, short drw_y, short vid_w, short vid_h,
|
|
|
|
short drw_w, short drw_h, RegionPtr clipBoxes,
|
|
|
|
pointer data, DrawablePtr pDraw)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidGetVideo:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidGetStill(ScrnInfoPtr pScrn, short vid_x, short vid_y,
|
|
|
|
short drw_x, short drw_y, short vid_w, short vid_h,
|
|
|
|
short drw_w, short drw_h, RegionPtr clipBoxes,
|
|
|
|
pointer data, DrawablePtr pDraw)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("FBDevTIVidGetStill:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void
|
|
|
|
xrdpVidStopVideo(ScrnInfoPtr pScrn, pointer data, Bool Cleanup)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidStopVideo:"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidSetPortAttribute(ScrnInfoPtr pScrn, Atom attribute,
|
|
|
|
INT32 value, pointer data)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidSetPortAttribute:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidGetPortAttribute(ScrnInfoPtr pScrn, Atom attribute,
|
|
|
|
INT32 *value, pointer data)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidGetPortAttribute:"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static void
|
|
|
|
xrdpVidQueryBestSize(ScrnInfoPtr pScrn, Bool motion,
|
|
|
|
short vid_w, short vid_h, short drw_w, short drw_h,
|
|
|
|
unsigned int *p_w, unsigned int *p_h, pointer data)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidQueryBestSize:"));
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
YV12_to_RGB32(unsigned char *yuvs, int width, int height, int *rgbs)
|
|
|
|
{
|
|
|
|
int size_total;
|
|
|
|
int y;
|
|
|
|
int u;
|
|
|
|
int v;
|
|
|
|
int c;
|
|
|
|
int d;
|
|
|
|
int e;
|
|
|
|
int r;
|
|
|
|
int g;
|
|
|
|
int b;
|
|
|
|
int t;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
size_total = width * height;
|
|
|
|
for (j = 0; j < height; j++)
|
|
|
|
{
|
|
|
|
for (i = 0; i < width; i++)
|
|
|
|
{
|
|
|
|
y = yuvs[j * width + i];
|
|
|
|
u = yuvs[(j / 2) * (width / 2) + (i / 2) + size_total];
|
|
|
|
v = yuvs[(j / 2) * (width / 2) + (i / 2) + size_total + (size_total / 4)];
|
|
|
|
c = y - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-21 07:10:20 +08:00
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
I420_to_RGB32(unsigned char *yuvs, int width, int height, int *rgbs)
|
|
|
|
{
|
|
|
|
int size_total;
|
|
|
|
int y;
|
|
|
|
int u;
|
|
|
|
int v;
|
|
|
|
int c;
|
|
|
|
int d;
|
|
|
|
int e;
|
|
|
|
int r;
|
|
|
|
int g;
|
|
|
|
int b;
|
|
|
|
int t;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
size_total = width * height;
|
|
|
|
for (j = 0; j < height; j++)
|
|
|
|
{
|
|
|
|
for (i = 0; i < width; i++)
|
|
|
|
{
|
|
|
|
y = yuvs[j * width + i];
|
|
|
|
v = yuvs[(j / 2) * (width / 2) + (i / 2) + size_total];
|
|
|
|
u = yuvs[(j / 2) * (width / 2) + (i / 2) + size_total + (size_total / 4)];
|
|
|
|
c = y - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
YUY2_to_RGB32(unsigned char *yuvs, int width, int height, int *rgbs)
|
|
|
|
{
|
|
|
|
int y1;
|
|
|
|
int y2;
|
|
|
|
int u;
|
|
|
|
int v;
|
|
|
|
int c;
|
|
|
|
int d;
|
|
|
|
int e;
|
|
|
|
int r;
|
|
|
|
int g;
|
|
|
|
int b;
|
|
|
|
int t;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j < height; j++)
|
|
|
|
{
|
|
|
|
for (i = 0; i < width; i++)
|
|
|
|
{
|
|
|
|
y1 = *(yuvs++);
|
|
|
|
v = *(yuvs++);
|
|
|
|
y2 = *(yuvs++);
|
|
|
|
u = *(yuvs++);
|
|
|
|
|
|
|
|
c = y1 - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
|
|
|
|
i++;
|
|
|
|
c = y2 - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
UYVY_to_RGB32(unsigned char *yuvs, int width, int height, int *rgbs)
|
|
|
|
{
|
|
|
|
int y1;
|
|
|
|
int y2;
|
|
|
|
int u;
|
|
|
|
int v;
|
|
|
|
int c;
|
|
|
|
int d;
|
|
|
|
int e;
|
|
|
|
int r;
|
|
|
|
int g;
|
|
|
|
int b;
|
|
|
|
int t;
|
|
|
|
int i;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
for (j = 0; j < height; j++)
|
|
|
|
{
|
|
|
|
for (i = 0; i < width; i++)
|
|
|
|
{
|
|
|
|
v = *(yuvs++);
|
|
|
|
y1 = *(yuvs++);
|
|
|
|
u = *(yuvs++);
|
|
|
|
y2 = *(yuvs++);
|
|
|
|
|
|
|
|
c = y1 - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
|
|
|
|
i++;
|
|
|
|
c = y2 - 16;
|
|
|
|
d = u - 128;
|
|
|
|
e = v - 128;
|
|
|
|
t = (298 * c + 409 * e + 128) >> 8;
|
|
|
|
b = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c - 100 * d - 208 * e + 128) >> 8;
|
|
|
|
g = RDPCLAMP(t, 0, 255);
|
|
|
|
t = (298 * c + 516 * d + 128) >> 8;
|
|
|
|
r = RDPCLAMP(t, 0, 255);
|
|
|
|
rgbs[j * width + i] = (r << 16) | (g << 8) | b;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-09-23 11:15:03 +08:00
|
|
|
#if 0
|
2014-09-21 04:21:37 +08:00
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
stretch_RGB32_RGB32(int *src, int src_width, int src_height,
|
|
|
|
int src_x, int src_y, int src_w, int src_h,
|
|
|
|
int *dst, int dst_w, int dst_h)
|
|
|
|
{
|
|
|
|
int mwidth;
|
|
|
|
int mheight;
|
|
|
|
int index;
|
|
|
|
|
|
|
|
mwidth = RDPMIN(src_width, dst_w);
|
|
|
|
mheight = RDPMIN(src_height, dst_h);
|
|
|
|
for (index = 0; index < mheight; index++)
|
|
|
|
{
|
|
|
|
g_memcpy(dst, src, mwidth * 4);
|
|
|
|
src += src_width;
|
|
|
|
dst += dst_w;
|
|
|
|
}
|
2014-09-23 11:15:03 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
stretch_RGB32_RGB32(int *src, int src_width, int src_height,
|
|
|
|
int src_x, int src_y, int src_w, int src_h,
|
|
|
|
int *dst, int dst_w, int dst_h)
|
|
|
|
{
|
|
|
|
int index;
|
|
|
|
int jndex;
|
|
|
|
int kndex;
|
|
|
|
int lndex;
|
|
|
|
int oh = (src_w << 16) / dst_w;
|
|
|
|
int ih;
|
|
|
|
int ov = (src_h << 16) / dst_h;
|
|
|
|
int iv;
|
|
|
|
int pix;
|
|
|
|
|
|
|
|
LLOGLN(10, ("stretch_RGB32_RGB32: oh 0x%8.8x ov 0x%8.8x", oh, ov));
|
|
|
|
iv = ov;
|
|
|
|
lndex = src_y;
|
|
|
|
for (index = 0; index < dst_h; index++)
|
|
|
|
{
|
|
|
|
ih = oh;
|
|
|
|
kndex = src_x;
|
|
|
|
for (jndex = 0; jndex < dst_w; jndex++)
|
|
|
|
{
|
|
|
|
pix = src[lndex * src_width + kndex];
|
|
|
|
dst[index * dst_w + jndex] = pix;
|
|
|
|
while (ih > (1 << 16) - 1)
|
|
|
|
{
|
|
|
|
ih -= 1 << 16;
|
|
|
|
kndex++;
|
|
|
|
}
|
|
|
|
ih += oh;
|
|
|
|
}
|
|
|
|
while (iv > (1 << 16) - 1)
|
|
|
|
{
|
|
|
|
iv -= 1 << 16;
|
|
|
|
lndex++;
|
|
|
|
}
|
|
|
|
iv += ov;
|
|
|
|
|
|
|
|
}
|
2014-09-21 04:21:37 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
2014-09-21 07:10:20 +08:00
|
|
|
xrdpVidPutImage(ScrnInfoPtr pScrn,
|
|
|
|
short src_x, short src_y, short drw_x, short drw_y,
|
|
|
|
short src_w, short src_h, short drw_w, short drw_h,
|
|
|
|
int format, unsigned char* buf,
|
|
|
|
short width, short height,
|
|
|
|
Bool sync, RegionPtr clipBoxes,
|
|
|
|
pointer data, DrawablePtr dst)
|
2014-09-21 04:21:37 +08:00
|
|
|
{
|
|
|
|
rdpPtr dev;
|
|
|
|
char *dst8;
|
|
|
|
int *dst32;
|
|
|
|
int *src32;
|
|
|
|
int *src32a;
|
|
|
|
int *rgborg32;
|
|
|
|
int *rgbend32;
|
|
|
|
int index;
|
|
|
|
int jndex;
|
|
|
|
int num_clips;
|
|
|
|
RegionRec dreg;
|
|
|
|
BoxRec box;
|
|
|
|
|
2014-09-21 07:10:20 +08:00
|
|
|
LLOGLN(10, ("xrdpVidPutImage:"));
|
2014-09-23 11:15:03 +08:00
|
|
|
LLOGLN(10, ("xrdpVidPutImage: src_x %d srcy_y %d", src_x, src_y));
|
2014-09-21 04:21:37 +08:00
|
|
|
dev = XRDPPTR(pScrn);
|
|
|
|
|
2014-09-21 07:10:20 +08:00
|
|
|
index = width * height * 4 + drw_w * drw_h * 4;
|
|
|
|
rgborg32 = (int *) g_malloc(index, 0);
|
|
|
|
if (rgborg32 == NULL)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("xrdpVidPutImage: memory alloc error"));
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
rgbend32 = rgborg32 + width * height;
|
2014-09-21 04:21:37 +08:00
|
|
|
|
2014-09-21 07:10:20 +08:00
|
|
|
switch (format)
|
|
|
|
{
|
|
|
|
case FOURCC_YV12:
|
|
|
|
LLOGLN(10, ("xrdpVidPutImage: FOURCC_YV12"));
|
|
|
|
YV12_to_RGB32(buf, width, height, rgborg32);
|
|
|
|
break;
|
|
|
|
case FOURCC_I420:
|
|
|
|
LLOGLN(10, ("xrdpVidPutImage: FOURCC_I420"));
|
|
|
|
I420_to_RGB32(buf, width, height, rgborg32);
|
|
|
|
break;
|
|
|
|
case FOURCC_YUY2:
|
|
|
|
LLOGLN(10, ("xrdpVidPutImage: FOURCC_YUY2"));
|
|
|
|
YUY2_to_RGB32(buf, width, height, rgborg32);
|
|
|
|
break;
|
|
|
|
case FOURCC_UYVY:
|
|
|
|
LLOGLN(10, ("xrdpVidPutImage: FOURCC_UYVY"));
|
|
|
|
UYVY_to_RGB32(buf, width, height, rgborg32);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LLOGLN(0, ("xrdpVidPutImage: unknown format 0x%8.8x", format));
|
|
|
|
g_free(rgborg32);
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
stretch_RGB32_RGB32(rgborg32, width, height,
|
|
|
|
src_x, src_y, src_w, src_h,
|
2014-09-21 04:21:37 +08:00
|
|
|
rgbend32, drw_w, drw_h);
|
|
|
|
|
|
|
|
box.x1 = drw_x;
|
|
|
|
box.y1 = drw_y;
|
|
|
|
box.x2 = box.x1 + drw_w;
|
|
|
|
box.y2 = box.y1 + drw_h;
|
2014-09-23 11:15:03 +08:00
|
|
|
LLOGLN(10, ("box 1 %d %d %d %d", box.x1, box.y1, box.x2, box.y2));
|
2014-09-21 04:21:37 +08:00
|
|
|
rdpRegionInit(&dreg, &box, 0);
|
|
|
|
|
|
|
|
num_clips = REGION_NUM_RECTS(clipBoxes);
|
2014-09-21 07:10:20 +08:00
|
|
|
LLOGLN(10, ("xrdpVidPutImage: num_clips %d", num_clips));
|
2014-09-21 04:21:37 +08:00
|
|
|
if (num_clips > 0)
|
|
|
|
{
|
|
|
|
rdpRegionIntersect(&dreg, &dreg, clipBoxes);
|
|
|
|
}
|
|
|
|
|
2014-09-21 07:57:53 +08:00
|
|
|
num_clips = REGION_NUM_RECTS(&dreg);
|
2014-09-21 04:21:37 +08:00
|
|
|
for (jndex = 0; jndex < num_clips; jndex++)
|
|
|
|
{
|
|
|
|
box = REGION_RECTS(&dreg)[jndex];
|
2014-09-23 11:15:03 +08:00
|
|
|
LLOGLN(10, ("box 2 %d %d %d %d", box.x1, box.y1, box.x2, box.y2));
|
2014-09-21 04:21:37 +08:00
|
|
|
dst8 = dev->pfbMemory + box.y1 * dev->paddedWidthInBytes;
|
|
|
|
src32a = rgbend32 + (box.y1 - drw_y) * drw_w;
|
|
|
|
for (index = 0; index < box.y2 - box.y1; index++)
|
|
|
|
{
|
|
|
|
dst32 = (int *) dst8;
|
|
|
|
dst32 += box.x1;
|
|
|
|
src32 = src32a + (box.x1 - drw_x);
|
|
|
|
g_memcpy(dst32, src32, (box.x2 - box.x1) * 4);
|
|
|
|
dst8 += dev->paddedWidthInBytes;
|
|
|
|
src32a += drw_w;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
rdpClientConAddAllReg(dev, &dreg, dst);
|
|
|
|
rdpRegionUninit(&dreg);
|
|
|
|
|
|
|
|
g_free(rgborg32);
|
|
|
|
|
|
|
|
return Success;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
static int
|
|
|
|
xrdpVidQueryImageAttributes(ScrnInfoPtr pScrn, int id,
|
|
|
|
unsigned short *w, unsigned short *h,
|
|
|
|
int *pitches, int *offsets)
|
|
|
|
{
|
|
|
|
int size, tmp;
|
|
|
|
|
2014-09-21 07:10:20 +08:00
|
|
|
LLOGLN(10, ("xrdpVidQueryImageAttributes:"));
|
2014-09-21 04:21:37 +08:00
|
|
|
/* this is same code as all drivers currently have */
|
|
|
|
if (*w > 2046)
|
|
|
|
{
|
|
|
|
*w = 2046;
|
|
|
|
}
|
|
|
|
if (*h > 2046)
|
|
|
|
{
|
|
|
|
*h = 2046;
|
|
|
|
}
|
|
|
|
/* make w multiple of 4 so that resizing works properly */
|
|
|
|
*w = (*w + 3) & ~3;
|
|
|
|
if (offsets != NULL)
|
|
|
|
{
|
|
|
|
offsets[0] = 0;
|
|
|
|
}
|
|
|
|
switch (id)
|
|
|
|
{
|
|
|
|
case FOURCC_YV12:
|
|
|
|
case FOURCC_I420:
|
|
|
|
/* make h be even */
|
|
|
|
*h = (*h + 1) & ~1;
|
|
|
|
/* make w be multiple of 4 (ie. pad it) */
|
|
|
|
size = (*w + 3) & ~3;
|
|
|
|
/* width of a Y row => width of image */
|
|
|
|
if (pitches != NULL)
|
|
|
|
{
|
|
|
|
pitches[0] = size;
|
|
|
|
}
|
|
|
|
/* offset of U plane => w * h */
|
|
|
|
size *= *h;
|
|
|
|
if (offsets != NULL)
|
|
|
|
{
|
|
|
|
offsets[1] = size;
|
|
|
|
}
|
|
|
|
/* width of U, V row => width / 2 */
|
|
|
|
tmp = ((*w >> 1) + 3) & ~3;
|
|
|
|
if (pitches != NULL)
|
|
|
|
{
|
|
|
|
pitches[1] = pitches[2] = tmp;
|
|
|
|
}
|
|
|
|
/* offset of V => Y plane + U plane (w * h + w / 2 * h / 2) */
|
|
|
|
tmp *= (*h >> 1);
|
|
|
|
size += tmp;
|
|
|
|
if (offsets != NULL)
|
|
|
|
{
|
|
|
|
offsets[2] = size;
|
|
|
|
}
|
|
|
|
size += tmp;
|
|
|
|
break;
|
|
|
|
case FOURCC_YUY2:
|
|
|
|
case FOURCC_UYVY:
|
|
|
|
size = (*w) * 2;
|
|
|
|
if (pitches != NULL)
|
|
|
|
{
|
|
|
|
pitches[0] = size;
|
|
|
|
}
|
|
|
|
size *= *h;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
LLOGLN(0, ("xrdpVidQueryImageAttributes: Unsupported image"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
LLOGLN(10, ("xrdpVidQueryImageAttributes: finished size %d id 0x%x", size, id));
|
|
|
|
return size;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*****************************************************************************/
|
|
|
|
Bool
|
|
|
|
rdpXvInit(ScreenPtr pScreen, ScrnInfoPtr pScrn)
|
|
|
|
{
|
|
|
|
XF86VideoAdaptorPtr adaptor;
|
|
|
|
DevUnion* pDevUnion;
|
|
|
|
int bytes;
|
|
|
|
|
|
|
|
adaptor = xf86XVAllocateVideoAdaptorRec(pScrn);
|
|
|
|
if (adaptor == 0)
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("rdpXvInit: xf86XVAllocateVideoAdaptorRec failed"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
adaptor->type = XvInputMask | XvImageMask | XvVideoMask | XvStillMask | XvWindowMask | XvPixmapMask;
|
2014-09-23 11:15:03 +08:00
|
|
|
//adaptor->flags = VIDEO_NO_CLIPPING;
|
|
|
|
//adaptor->flags = VIDEO_CLIP_TO_VIEWPORT;
|
|
|
|
adaptor->flags = 0;
|
2014-09-21 16:28:29 +08:00
|
|
|
adaptor->name = XRDP_MODULE_NAME " XVideo Adaptor";
|
2014-09-21 04:21:37 +08:00
|
|
|
adaptor->nEncodings = T_NUM_ENCODINGS;
|
|
|
|
adaptor->pEncodings = &(g_xrdpVidEncodings[0]);
|
|
|
|
adaptor->nFormats = T_NUM_FORMATS;
|
|
|
|
adaptor->pFormats = &(g_xrdpVidFormats[0]);
|
|
|
|
adaptor->pFormats[0].depth = pScrn->depth;
|
|
|
|
LLOGLN(0, ("rdpXvInit: depth %d", pScrn->depth));
|
|
|
|
adaptor->nImages = sizeof(g_xrdpVidImages) / sizeof(XF86ImageRec);
|
|
|
|
adaptor->pImages = g_xrdpVidImages;
|
|
|
|
adaptor->nAttributes = 0;
|
|
|
|
adaptor->pAttributes = 0;
|
|
|
|
adaptor->nPorts = T_MAX_PORTS;
|
|
|
|
bytes = sizeof(DevUnion) * T_MAX_PORTS;
|
|
|
|
pDevUnion = (DevUnion*) g_malloc(bytes, 1);
|
|
|
|
adaptor->pPortPrivates = pDevUnion;
|
|
|
|
adaptor->PutVideo = xrdpVidPutVideo;
|
|
|
|
adaptor->PutStill = xrdpVidPutStill;
|
|
|
|
adaptor->GetVideo = xrdpVidGetVideo;
|
|
|
|
adaptor->GetStill = xrdpVidGetStill;
|
|
|
|
adaptor->StopVideo = xrdpVidStopVideo;
|
|
|
|
adaptor->SetPortAttribute = xrdpVidSetPortAttribute;
|
|
|
|
adaptor->GetPortAttribute = xrdpVidGetPortAttribute;
|
|
|
|
adaptor->QueryBestSize = xrdpVidQueryBestSize;
|
|
|
|
adaptor->PutImage = xrdpVidPutImage;
|
|
|
|
adaptor->QueryImageAttributes = xrdpVidQueryImageAttributes;
|
|
|
|
if (!xf86XVScreenInit(pScreen, &adaptor, 1))
|
|
|
|
{
|
|
|
|
LLOGLN(0, ("rdpXvInit: xf86XVScreenInit failed"));
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
xf86XVFreeVideoAdaptorRec(adaptor);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|