mirror of
https://github.com/hyperion-project/hyperion.ng.git
synced 2025-03-01 10:33:28 +00:00
Feature/xcb grabber (#912)
* Add Xcb grabber * update compile instruction Signed-off-by: Paulchen Panther <Paulchen-Panter@protonmail.com> * Fix problem on resolution change + Make XCB default if X11 is not avaialable * Fix decimation problem Co-authored-by: Paulchen Panther <16664240+Paulchen-Panther@users.noreply.github.com> Co-authored-by: Paulchen Panther <Paulchen-Panter@protonmail.com>
This commit is contained in:
@@ -22,6 +22,10 @@ if (ENABLE_X11)
|
||||
add_subdirectory(x11)
|
||||
endif()
|
||||
|
||||
if (ENABLE_XCB)
|
||||
add_subdirectory(xcb)
|
||||
endif()
|
||||
|
||||
if (ENABLE_QT)
|
||||
add_subdirectory(qt)
|
||||
endif()
|
||||
|
@@ -72,8 +72,13 @@ void X11Grabber::setupResources()
|
||||
_shminfo.readOnly = False;
|
||||
XShmAttach(_x11Display, &_shminfo);
|
||||
}
|
||||
|
||||
if (_XRenderAvailable)
|
||||
{
|
||||
_useImageResampler = false;
|
||||
_imageResampler.setHorizontalPixelDecimation(1);
|
||||
_imageResampler.setVerticalPixelDecimation(1);
|
||||
|
||||
if(_XShmPixmapAvailable)
|
||||
{
|
||||
_pixmap = XShmCreatePixmap(_x11Display, _window, _xImage->data, &_shminfo, _width, _height, _windowAttr.depth);
|
||||
@@ -86,8 +91,16 @@ void X11Grabber::setupResources()
|
||||
_dstFormat = XRenderFindVisualFormat(_x11Display, _windowAttr.visual);
|
||||
_srcPicture = XRenderCreatePicture(_x11Display, _window, _srcFormat, CPRepeat, &_pictAttr);
|
||||
_dstPicture = XRenderCreatePicture(_x11Display, _pixmap, _dstFormat, CPRepeat, &_pictAttr);
|
||||
|
||||
XRenderSetPictureFilter(_x11Display, _srcPicture, FilterBilinear, NULL, 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
_useImageResampler = true;
|
||||
_imageResampler.setHorizontalPixelDecimation(_pixelDecimation);
|
||||
_imageResampler.setVerticalPixelDecimation(_pixelDecimation);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
bool X11Grabber::Setup()
|
||||
@@ -117,10 +130,6 @@ bool X11Grabber::Setup()
|
||||
XShmQueryVersion(_x11Display, &dummy, &dummy, &pixmaps_supported);
|
||||
_XShmPixmapAvailable = pixmaps_supported && XShmPixmapFormat(_x11Display) == ZPixmap;
|
||||
|
||||
// Image scaling is performed by XRender when available, otherwise by ImageResampler
|
||||
_imageResampler.setHorizontalPixelDecimation(_XRenderAvailable ? 1 : _pixelDecimation);
|
||||
_imageResampler.setVerticalPixelDecimation(_XRenderAvailable ? 1 : _pixelDecimation);
|
||||
|
||||
bool result = (updateScreenDimensions(true) >=0);
|
||||
ErrorIf(!result, _log, "X11 Grabber start failed");
|
||||
setEnabled(result);
|
||||
|
@@ -23,13 +23,17 @@ void X11Wrapper::action()
|
||||
{
|
||||
stop();
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_grabber.updateScreenDimensions() < 0 )
|
||||
{
|
||||
stop();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (isActive())
|
||||
{
|
||||
if (_grabber.updateScreenDimensions() >= 0 )
|
||||
{
|
||||
transferFrame(_grabber);
|
||||
}
|
||||
transferFrame(_grabber);
|
||||
}
|
||||
}
|
||||
|
21
libsrc/grabber/xcb/CMakeLists.txt
Normal file
21
libsrc/grabber/xcb/CMakeLists.txt
Normal file
@@ -0,0 +1,21 @@
|
||||
# Define the current source locations
|
||||
SET(CURRENT_HEADER_DIR ${CMAKE_SOURCE_DIR}/include/grabber)
|
||||
SET(CURRENT_SOURCE_DIR ${CMAKE_SOURCE_DIR}/libsrc/grabber/xcb)
|
||||
|
||||
find_package(XCB COMPONENTS SHM IMAGE RENDER RANDR REQUIRED)
|
||||
find_package(Qt5Widgets REQUIRED)
|
||||
find_package(Qt5X11Extras REQUIRED)
|
||||
|
||||
include_directories(${XCB_INCLUDE_DIRS})
|
||||
|
||||
FILE (GLOB XCB_SOURCES "${CURRENT_HEADER_DIR}/Xcb*.h" "${CURRENT_SOURCE_DIR}/*.h" "${CURRENT_SOURCE_DIR}/*.cpp" )
|
||||
|
||||
add_library(xcb-grabber ${XCB_SOURCES})
|
||||
|
||||
target_link_libraries(xcb-grabber
|
||||
hyperion
|
||||
Qt5::X11Extras
|
||||
Qt5::Widgets
|
||||
${XCB_LIBRARIES}
|
||||
)
|
||||
|
30
libsrc/grabber/xcb/XcbCommandExecutor.h
Normal file
30
libsrc/grabber/xcb/XcbCommandExecutor.h
Normal file
@@ -0,0 +1,30 @@
|
||||
#pragma once
|
||||
|
||||
#include <memory>
|
||||
|
||||
#include <xcb/xcb.h>
|
||||
|
||||
template<class Request, class ...Args>
|
||||
std::unique_ptr<typename Request::ResponseType, decltype(&free)>
|
||||
query(xcb_connection_t * connection, Args&& ...args)
|
||||
{
|
||||
auto cookie = Request::RequestFunction(connection,args...);
|
||||
|
||||
xcb_generic_error_t * error = nullptr;
|
||||
std::unique_ptr<typename Request::ResponseType, decltype(&free)> xcbResponse(
|
||||
Request::ReplyFunction(connection, cookie, &error), free);
|
||||
|
||||
if (error) {
|
||||
Logger * LOGGER = Logger::getInstance("XCB");
|
||||
Error(LOGGER,
|
||||
"Cannot get the image data event_error: response_type:%u error_code:%u "
|
||||
"sequence:%u resource_id:%u minor_code:%u major_code:%u.\n",
|
||||
error->response_type, error->error_code, error->sequence,
|
||||
error->resource_id, error->minor_code, error->major_code);
|
||||
|
||||
free(error);
|
||||
return {nullptr, nullptr};
|
||||
}
|
||||
|
||||
return xcbResponse;
|
||||
}
|
55
libsrc/grabber/xcb/XcbCommands.h
Normal file
55
libsrc/grabber/xcb/XcbCommands.h
Normal file
@@ -0,0 +1,55 @@
|
||||
#pragma once
|
||||
|
||||
#include <xcb/randr.h>
|
||||
#include <xcb/shm.h>
|
||||
#include <xcb/xcb.h>
|
||||
#include <xcb/xcb_image.h>
|
||||
|
||||
struct GetImage
|
||||
{
|
||||
typedef xcb_get_image_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_get_image;
|
||||
static constexpr auto ReplyFunction = xcb_get_image_reply;
|
||||
};
|
||||
|
||||
struct GetGeometry
|
||||
{
|
||||
typedef xcb_get_geometry_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_get_geometry;
|
||||
static constexpr auto ReplyFunction = xcb_get_geometry_reply;
|
||||
};
|
||||
|
||||
struct ShmQueryVersion
|
||||
{
|
||||
typedef xcb_shm_query_version_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_shm_query_version;
|
||||
static constexpr auto ReplyFunction = xcb_shm_query_version_reply;
|
||||
};
|
||||
|
||||
struct RenderQueryVersion
|
||||
{
|
||||
typedef xcb_render_query_version_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_render_query_version;
|
||||
static constexpr auto ReplyFunction = xcb_render_query_version_reply;
|
||||
};
|
||||
|
||||
struct ShmGetImage
|
||||
{
|
||||
typedef xcb_shm_get_image_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_shm_get_image;
|
||||
static constexpr auto ReplyFunction = xcb_shm_get_image_reply;
|
||||
};
|
||||
|
||||
struct RenderQueryPictFormats
|
||||
{
|
||||
typedef xcb_render_query_pict_formats_reply_t ResponseType;
|
||||
|
||||
static constexpr auto RequestFunction = xcb_render_query_pict_formats;
|
||||
static constexpr auto ReplyFunction = xcb_render_query_pict_formats_reply;
|
||||
};
|
||||
|
452
libsrc/grabber/xcb/XcbGrabber.cpp
Normal file
452
libsrc/grabber/xcb/XcbGrabber.cpp
Normal file
@@ -0,0 +1,452 @@
|
||||
#include <utils/Logger.h>
|
||||
#include <grabber/XcbGrabber.h>
|
||||
|
||||
#include "XcbCommands.h"
|
||||
#include "XcbCommandExecutor.h"
|
||||
|
||||
#include <xcb/xcb_event.h>
|
||||
|
||||
#include <QCoreApplication>
|
||||
#include <QX11Info>
|
||||
|
||||
#include <memory>
|
||||
|
||||
#define DOUBLE_TO_FIXED(d) ((xcb_render_fixed_t) ((d) * 65536))
|
||||
|
||||
XcbGrabber::XcbGrabber(int cropLeft, int cropRight, int cropTop, int cropBottom, int pixelDecimation)
|
||||
: Grabber("XCBGRABBER", 0, 0, cropLeft, cropRight, cropTop, cropBottom)
|
||||
, _connection{}
|
||||
, _screen{}
|
||||
, _pixmap{}
|
||||
, _srcFormat{}
|
||||
, _dstFormat{}
|
||||
, _srcPicture{}
|
||||
, _dstPicture{}
|
||||
, _transform{}
|
||||
, _shminfo{}
|
||||
, _pixelDecimation(pixelDecimation)
|
||||
, _screenWidth{}
|
||||
, _screenHeight{}
|
||||
, _src_x(cropLeft)
|
||||
, _src_y(cropTop)
|
||||
, _XcbRenderAvailable{}
|
||||
, _XcbRandRAvailable{}
|
||||
, _XcbShmAvailable{}
|
||||
, _XcbShmPixmapAvailable{}
|
||||
, _logger{}
|
||||
, _shmData{}
|
||||
, _XcbRandREventBase{-1}
|
||||
{
|
||||
_logger = Logger::getInstance("XCB");
|
||||
|
||||
// cropping is performed by XcbRender, XcbShmGetImage or XcbGetImage
|
||||
_useImageResampler = false;
|
||||
_imageResampler.setCropping(0, 0, 0, 0);
|
||||
}
|
||||
|
||||
XcbGrabber::~XcbGrabber()
|
||||
{
|
||||
if (_connection != nullptr)
|
||||
{
|
||||
freeResources();
|
||||
xcb_disconnect(_connection);
|
||||
}
|
||||
}
|
||||
|
||||
void XcbGrabber::freeResources()
|
||||
{
|
||||
if (_XcbRandRAvailable)
|
||||
{
|
||||
qApp->removeNativeEventFilter(this);
|
||||
}
|
||||
|
||||
if(_XcbShmAvailable)
|
||||
{
|
||||
xcb_shm_detach(_connection, _shminfo);
|
||||
shmdt(_shmData);
|
||||
shmctl(_shminfo, IPC_RMID, 0);
|
||||
|
||||
}
|
||||
|
||||
if (_XcbRenderAvailable)
|
||||
{
|
||||
xcb_free_pixmap(_connection, _pixmap);
|
||||
xcb_render_free_picture(_connection, _srcPicture);
|
||||
xcb_render_free_picture(_connection, _dstPicture);
|
||||
}
|
||||
}
|
||||
|
||||
void XcbGrabber::setupResources()
|
||||
{
|
||||
if (_XcbRandRAvailable)
|
||||
{
|
||||
qApp->installNativeEventFilter(this);
|
||||
}
|
||||
|
||||
if(_XcbShmAvailable)
|
||||
{
|
||||
_shminfo = xcb_generate_id(_connection);
|
||||
int id = shmget(IPC_PRIVATE, _width * _height * 4, IPC_CREAT | 0777);
|
||||
_shmData = static_cast<uint8_t*>(shmat(id, nullptr, 0));
|
||||
xcb_shm_attach(_connection, _shminfo, id, 0);
|
||||
}
|
||||
|
||||
if (_XcbRenderAvailable)
|
||||
{
|
||||
_useImageResampler = false;
|
||||
_imageResampler.setHorizontalPixelDecimation(1);
|
||||
_imageResampler.setVerticalPixelDecimation(1);
|
||||
|
||||
if(_XcbShmPixmapAvailable)
|
||||
{
|
||||
_pixmap = xcb_generate_id(_connection);
|
||||
xcb_shm_create_pixmap(
|
||||
_connection, _pixmap, _screen->root, _width,
|
||||
_height, _screen->root_depth, _shminfo, 0);
|
||||
}
|
||||
else
|
||||
{
|
||||
_pixmap = xcb_generate_id(_connection);
|
||||
xcb_create_pixmap(_connection, _screen->root_depth, _pixmap, _screen->root, _width, _height);
|
||||
}
|
||||
|
||||
_srcFormat = findFormatForVisual(_screen->root_visual);
|
||||
_dstFormat = findFormatForVisual(_screen->root_visual);
|
||||
|
||||
_srcPicture = xcb_generate_id(_connection);
|
||||
_dstPicture = xcb_generate_id(_connection);
|
||||
|
||||
const uint32_t value_mask = XCB_RENDER_CP_REPEAT;
|
||||
const uint32_t values[] = { XCB_RENDER_REPEAT_NONE };
|
||||
|
||||
xcb_render_create_picture(_connection, _srcPicture, _screen->root, _srcFormat, value_mask, values);
|
||||
xcb_render_create_picture(_connection, _dstPicture, _pixmap, _dstFormat, value_mask, values);
|
||||
|
||||
const std::string filter = "fast";
|
||||
xcb_render_set_picture_filter(_connection, _srcPicture, filter.size(), filter.c_str(), 0, nullptr);
|
||||
}
|
||||
else
|
||||
{
|
||||
_useImageResampler = true;
|
||||
_imageResampler.setHorizontalPixelDecimation(_pixelDecimation);
|
||||
_imageResampler.setVerticalPixelDecimation(_pixelDecimation);
|
||||
}
|
||||
}
|
||||
|
||||
xcb_screen_t * XcbGrabber::getScreen(const xcb_setup_t *setup, int screen_num) const
|
||||
{
|
||||
xcb_screen_iterator_t it = xcb_setup_roots_iterator(setup);
|
||||
xcb_screen_t * screen = nullptr;
|
||||
|
||||
for (; it.rem > 0; xcb_screen_next(&it))
|
||||
{
|
||||
if (!screen_num)
|
||||
{
|
||||
screen = it.data;
|
||||
break;
|
||||
}
|
||||
screen_num--;
|
||||
}
|
||||
return screen;
|
||||
}
|
||||
|
||||
void XcbGrabber::setupRandr()
|
||||
{
|
||||
auto randrQueryExtensionReply = xcb_get_extension_data(_connection, &xcb_randr_id);
|
||||
_XcbRandRAvailable = randrQueryExtensionReply != nullptr;
|
||||
_XcbRandREventBase = randrQueryExtensionReply ? randrQueryExtensionReply->first_event : -1;
|
||||
}
|
||||
|
||||
void XcbGrabber::setupRender()
|
||||
{
|
||||
auto renderQueryVersionReply = query<RenderQueryVersion>(_connection, 0, 0);
|
||||
|
||||
_XcbRenderAvailable = renderQueryVersionReply != nullptr;
|
||||
}
|
||||
|
||||
void XcbGrabber::setupShm()
|
||||
{
|
||||
auto shmQueryExtensionReply = xcb_get_extension_data(_connection, &xcb_render_id);
|
||||
_XcbShmAvailable = shmQueryExtensionReply != nullptr;
|
||||
|
||||
_XcbShmPixmapAvailable = false;
|
||||
if (_XcbShmAvailable)
|
||||
{
|
||||
auto shmQueryVersionReply = query<ShmQueryVersion>(_connection);
|
||||
|
||||
_XcbShmPixmapAvailable = shmQueryVersionReply ? shmQueryVersionReply->shared_pixmaps : false;
|
||||
}
|
||||
}
|
||||
|
||||
bool XcbGrabber::Setup()
|
||||
{
|
||||
int screen_num;
|
||||
_connection = xcb_connect(nullptr, &screen_num);
|
||||
|
||||
int ret = xcb_connection_has_error(_connection);
|
||||
if (ret != 0)
|
||||
{
|
||||
Error(_logger, "Cannot open display, error %d", ret);
|
||||
return false;
|
||||
}
|
||||
|
||||
const xcb_setup_t * setup = xcb_get_setup(_connection);
|
||||
_screen = getScreen(setup, screen_num);
|
||||
|
||||
if (!_screen)
|
||||
{
|
||||
Error(_log, "Unable to open display, screen %d does not exist", screen_num);
|
||||
|
||||
if (getenv("DISPLAY"))
|
||||
Error(_log, "%s", getenv("DISPLAY"));
|
||||
else
|
||||
Error(_log, "DISPLAY environment variable not set");
|
||||
|
||||
freeResources();
|
||||
return false;
|
||||
}
|
||||
|
||||
setupRandr();
|
||||
setupRender();
|
||||
setupShm();
|
||||
|
||||
Info(_log, "XcbRandR : %s", _XcbRandRAvailable ? "available" : "unavailable");
|
||||
Info(_log, "XcbRender : %s", _XcbRenderAvailable ? "available" : "unavailable");
|
||||
Info(_log, "XcbShm : %s", _XcbShmAvailable ? "available" : "unavailable");
|
||||
Info(_log, "XcbPixmap : %s", _XcbShmPixmapAvailable ? "available" : "unavailable");
|
||||
|
||||
bool result = (updateScreenDimensions(true) >=0);
|
||||
ErrorIf(!result, _log, "XCB Grabber start failed");
|
||||
setEnabled(result);
|
||||
return result;
|
||||
}
|
||||
|
||||
int XcbGrabber::grabFrame(Image<ColorRgb> & image, bool forceUpdate)
|
||||
{
|
||||
if (!_enabled)
|
||||
return 0;
|
||||
|
||||
if (forceUpdate)
|
||||
updateScreenDimensions(forceUpdate);
|
||||
|
||||
if (_XcbRenderAvailable)
|
||||
{
|
||||
double scale_x = static_cast<double>(_screenWidth / _pixelDecimation) / static_cast<double>(_screenWidth);
|
||||
double scale_y = static_cast<double>(_screenHeight / _pixelDecimation) / static_cast<double>(_screenHeight);
|
||||
double scale = qMin(scale_y, scale_x);
|
||||
|
||||
_transform = {
|
||||
DOUBLE_TO_FIXED(1), DOUBLE_TO_FIXED(0), DOUBLE_TO_FIXED(0),
|
||||
DOUBLE_TO_FIXED(0), DOUBLE_TO_FIXED(1), DOUBLE_TO_FIXED(0),
|
||||
DOUBLE_TO_FIXED(0), DOUBLE_TO_FIXED(0), DOUBLE_TO_FIXED(scale)
|
||||
};
|
||||
|
||||
xcb_render_set_picture_transform(_connection, _srcPicture, _transform);
|
||||
xcb_render_composite(_connection,
|
||||
XCB_RENDER_PICT_OP_SRC, _srcPicture,
|
||||
XCB_RENDER_PICTURE_NONE, _dstPicture,
|
||||
(_src_x/_pixelDecimation),
|
||||
(_src_y/_pixelDecimation),
|
||||
0, 0, 0, 0, _width, _height);
|
||||
|
||||
xcb_flush(_connection);
|
||||
|
||||
if (_XcbShmAvailable)
|
||||
{
|
||||
query<ShmGetImage>(_connection,
|
||||
_pixmap, 0, 0, _width, _height,
|
||||
~0, XCB_IMAGE_FORMAT_Z_PIXMAP, _shminfo, 0);
|
||||
|
||||
_imageResampler.processImage(
|
||||
reinterpret_cast<const uint8_t *>(_shmData),
|
||||
_width, _height, _width * 4, PixelFormat::BGR32, image);
|
||||
}
|
||||
else
|
||||
{
|
||||
auto result = query<GetImage>(_connection,
|
||||
XCB_IMAGE_FORMAT_Z_PIXMAP, _pixmap,
|
||||
0, 0, _width, _height, ~0);
|
||||
|
||||
auto buffer = xcb_get_image_data(result.get());
|
||||
|
||||
_imageResampler.processImage(
|
||||
reinterpret_cast<const uint8_t *>(buffer),
|
||||
_width, _height, _width * 4, PixelFormat::BGR32, image);
|
||||
}
|
||||
|
||||
}
|
||||
else if (_XcbShmAvailable)
|
||||
{
|
||||
query<ShmGetImage>(_connection,
|
||||
_screen->root, _src_x, _src_y, _width, _height,
|
||||
~0, XCB_IMAGE_FORMAT_Z_PIXMAP, _shminfo, 0);
|
||||
|
||||
_imageResampler.processImage(
|
||||
reinterpret_cast<const uint8_t *>(_shmData),
|
||||
_width, _height, _width * 4, PixelFormat::BGR32, image);
|
||||
}
|
||||
else
|
||||
{
|
||||
auto result = query<GetImage>(_connection,
|
||||
XCB_IMAGE_FORMAT_Z_PIXMAP, _screen->root,
|
||||
_src_x, _src_y, _width, _height, ~0);
|
||||
|
||||
auto buffer = xcb_get_image_data(result.get());
|
||||
|
||||
_imageResampler.processImage(
|
||||
reinterpret_cast<const uint8_t *>(buffer),
|
||||
_width, _height, _width * 4, PixelFormat::BGR32, image);
|
||||
}
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
int XcbGrabber::updateScreenDimensions(bool force)
|
||||
{
|
||||
auto geometry = query<GetGeometry>(_connection, _screen->root);
|
||||
if (geometry == nullptr)
|
||||
{
|
||||
setEnabled(false);
|
||||
Error(_log, "Failed to obtain screen geometry");
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!_enabled)
|
||||
setEnabled(true);
|
||||
|
||||
if (!force && _screenWidth == unsigned(geometry->width) &&
|
||||
_screenHeight == unsigned(geometry->height))
|
||||
return 0;
|
||||
|
||||
if (_screenWidth || _screenHeight)
|
||||
freeResources();
|
||||
|
||||
Info(_log, "Update of screen resolution: [%dx%d] to [%dx%d]", _screenWidth, _screenHeight, geometry->width, geometry->height);
|
||||
|
||||
_screenWidth = geometry->width;
|
||||
_screenHeight = geometry->height;
|
||||
|
||||
int width = 0, height = 0;
|
||||
|
||||
// Image scaling is performed by XRender when available, otherwise by ImageResampler
|
||||
if (_XcbRenderAvailable)
|
||||
{
|
||||
width = (_screenWidth > unsigned(_cropLeft + _cropRight))
|
||||
? ((_screenWidth - _cropLeft - _cropRight) / _pixelDecimation)
|
||||
: _screenWidth / _pixelDecimation;
|
||||
|
||||
height = (_screenHeight > unsigned(_cropTop + _cropBottom))
|
||||
? ((_screenHeight - _cropTop - _cropBottom) / _pixelDecimation)
|
||||
: _screenHeight / _pixelDecimation;
|
||||
|
||||
Info(_log, "Using XcbRender for grabbing [%dx%d]", width, height);
|
||||
}
|
||||
else
|
||||
{
|
||||
width = (_screenWidth > unsigned(_cropLeft + _cropRight))
|
||||
? (_screenWidth - _cropLeft - _cropRight)
|
||||
: _screenWidth;
|
||||
|
||||
height = (_screenHeight > unsigned(_cropTop + _cropBottom))
|
||||
? (_screenHeight - _cropTop - _cropBottom)
|
||||
: _screenHeight;
|
||||
|
||||
Info(_log, "Using XcbGetImage for grabbing [%dx%d]", width, height);
|
||||
}
|
||||
|
||||
// Calculate final image dimensions and adjust top/left cropping in 3D modes
|
||||
switch (_videoMode)
|
||||
{
|
||||
case VideoMode::VIDEO_3DSBS:
|
||||
_width = width /2;
|
||||
_height = height;
|
||||
_src_x = _cropLeft / 2;
|
||||
_src_y = _cropTop;
|
||||
break;
|
||||
case VideoMode::VIDEO_3DTAB:
|
||||
_width = width;
|
||||
_height = height / 2;
|
||||
_src_x = _cropLeft;
|
||||
_src_y = _cropTop / 2;
|
||||
break;
|
||||
case VideoMode::VIDEO_2D:
|
||||
default:
|
||||
_width = width;
|
||||
_height = height;
|
||||
_src_x = _cropLeft;
|
||||
_src_y = _cropTop;
|
||||
break;
|
||||
}
|
||||
|
||||
setupResources();
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
void XcbGrabber::setVideoMode(VideoMode mode)
|
||||
{
|
||||
Grabber::setVideoMode(mode);
|
||||
updateScreenDimensions(true);
|
||||
}
|
||||
|
||||
void XcbGrabber::setPixelDecimation(int pixelDecimation)
|
||||
{
|
||||
if(_pixelDecimation != pixelDecimation)
|
||||
{
|
||||
_pixelDecimation = pixelDecimation;
|
||||
updateScreenDimensions(true);
|
||||
}
|
||||
}
|
||||
|
||||
void XcbGrabber::setCropping(unsigned cropLeft, unsigned cropRight, unsigned cropTop, unsigned cropBottom)
|
||||
{
|
||||
Grabber::setCropping(cropLeft, cropRight, cropTop, cropBottom);
|
||||
if(_connection != nullptr)
|
||||
updateScreenDimensions(true);
|
||||
}
|
||||
|
||||
bool XcbGrabber::nativeEventFilter(const QByteArray & eventType, void * message, long int * /*result*/)
|
||||
{
|
||||
if (!_XcbRandRAvailable || eventType != "xcb_generic_event_t" || _XcbRandREventBase == -1)
|
||||
return false;
|
||||
|
||||
xcb_generic_event_t *e = static_cast<xcb_generic_event_t*>(message);
|
||||
const uint8_t xEventType = XCB_EVENT_RESPONSE_TYPE(e);
|
||||
|
||||
if (xEventType == _XcbRandREventBase + XCB_RANDR_SCREEN_CHANGE_NOTIFY)
|
||||
updateScreenDimensions(true);
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
xcb_render_pictformat_t XcbGrabber::findFormatForVisual(xcb_visualid_t visual) const
|
||||
{
|
||||
auto formats = query<RenderQueryPictFormats>(_connection);
|
||||
if (formats == nullptr)
|
||||
return {};
|
||||
|
||||
int screen = QX11Info::appScreen();
|
||||
xcb_render_pictscreen_iterator_t sit =
|
||||
xcb_render_query_pict_formats_screens_iterator(formats.get());
|
||||
|
||||
for (; sit.rem; --screen, xcb_render_pictscreen_next(&sit)) {
|
||||
if (screen != 0)
|
||||
continue;
|
||||
|
||||
xcb_render_pictdepth_iterator_t dit =
|
||||
xcb_render_pictscreen_depths_iterator(sit.data);
|
||||
for (; dit.rem; xcb_render_pictdepth_next(&dit))
|
||||
{
|
||||
xcb_render_pictvisual_iterator_t vit
|
||||
= xcb_render_pictdepth_visuals_iterator(dit.data);
|
||||
for (; vit.rem; xcb_render_pictvisual_next(&vit))
|
||||
{
|
||||
if (vit.data->visual == visual)
|
||||
{
|
||||
return vit.data->format;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
return {};
|
||||
}
|
39
libsrc/grabber/xcb/XcbWrapper.cpp
Normal file
39
libsrc/grabber/xcb/XcbWrapper.cpp
Normal file
@@ -0,0 +1,39 @@
|
||||
#include <grabber/XcbWrapper.h>
|
||||
|
||||
XcbWrapper::XcbWrapper(int cropLeft, int cropRight, int cropTop, int cropBottom, int pixelDecimation, const unsigned updateRate_Hz)
|
||||
: GrabberWrapper("Xcb", &_grabber, 0, 0, updateRate_Hz)
|
||||
, _grabber(cropLeft, cropRight, cropTop, cropBottom, pixelDecimation)
|
||||
, _init(false)
|
||||
{}
|
||||
|
||||
XcbWrapper::~XcbWrapper()
|
||||
{
|
||||
if ( _init )
|
||||
{
|
||||
stop();
|
||||
}
|
||||
}
|
||||
|
||||
void XcbWrapper::action()
|
||||
{
|
||||
if (! _init )
|
||||
{
|
||||
_init = true;
|
||||
if ( ! _grabber.Setup() )
|
||||
{
|
||||
stop();
|
||||
}
|
||||
else
|
||||
{
|
||||
if (_grabber.updateScreenDimensions() < 0 )
|
||||
{
|
||||
stop();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if (isActive())
|
||||
{
|
||||
transferFrame(_grabber);
|
||||
}
|
||||
}
|
Reference in New Issue
Block a user