Skip to content

Commit

Permalink
Support shm key and drm for rawfb option
Browse files Browse the repository at this point in the history
This will support both x11 and Wayland. (#227)

* Add support shm key for rawfb option

Sometime shmid is not same on different time or system, we should use a constant ID for shm. Here use shm key to instead.

* Add DRM support

access GPU with DRM, this will support x11 and wayland

* Fix a bug for shmget params.

shmget should be 0 otherwise it would alloc a new shared memory but not for the frame buffer.

* fix drm configure while compiling.
  • Loading branch information
linex-cd authored Aug 1, 2023
1 parent 354602c commit a61f364
Show file tree
Hide file tree
Showing 5 changed files with 161 additions and 3 deletions.
22 changes: 22 additions & 0 deletions configure.ac
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,12 @@ AC_ARG_WITH(ssl,
[ --without-ssl disable support for openssl libssl]
[ --with-ssl=DIR use openssl include/library files in DIR],,)

AH_TEMPLATE(HAVE_LIBDRM, [drm library present])
AC_ARG_WITH(drm,
[ --without-drm disable support for library libdrm]
[ --with-drm=DIR use drm include/library files in DIR],,)


if test "x$with_crypto" != "xno" -a "x$with_ssl" != "xno"; then
if test ! -z "$with_ssl" -a "x$with_ssl" != "xyes"; then
saved_CPPFLAGS="$CPPFLAGS"
Expand Down Expand Up @@ -59,6 +65,22 @@ if test "x$with_crypt" != "xno"; then
fi
AC_SUBST(CRYPT_LIBS)

AH_TEMPLATE(HAVE_LIBDRM, [libdrm library present])
AC_ARG_WITH(drm,
[ --without-drm disable support for libdrm],,)
if test "x$with_drm" != "xno"; then
AC_CHECK_FUNCS([drm], HAVE_LIBC_DRM="true")
if test -z "$HAVE_LIBC_DRM"; then
AC_CHECK_LIB(drm, drmModeGetResources,
DRM_LIBS="-ldrm"
[AC_DEFINE(HAVE_LIBDRM) HAVE_LIBDRM="true"], ,)


CPPFLAGS="$saved_CPPFLAGS -I/usr/include/drm/"
fi
fi
AC_SUBST(DRM_LIBS)

AH_TEMPLATE(HAVE_X509_PRINT_EX_FP, [open ssl X509_print_ex_fp available])
if test "x$with_ssl" != "xno"; then
if test "x$HAVE_LIBCRYPTO" = "xtrue"; then
Expand Down
11 changes: 10 additions & 1 deletion doc/OPTIONS.md
Original file line number Diff line number Diff line change
Expand Up @@ -4395,8 +4395,16 @@ Options:
whole frame).
For shared memory segments string is of the form:
"shm:N@WxHxB" which specifies a shmid N and with
"shm:N@WxHxB" which specifies a shm key(or a shmid) N and with
WxHxB as above. See shmat(1) and ipcs(1)
Since shm key can be defined constantly, you should use
shm key rather than shmid. Unless you have a special purpose,
a mutable shmid may cause problems.
If there is a new desktop like Wayland, use "drm:" to access
GPU FrameBuffer. If you do not know the GPU path, just use
"/dev/dri/card0@WxHxB".
If you do not supply a type "map" is assumed if
the file exists (see the next paragraphs for some
Expand Down Expand Up @@ -4447,6 +4455,7 @@ Options:
Examples:
-rawfb shm:210337933@800x600x32:ff/ff00/ff0000
-rawfb drm:/dev/dri/card0@800x600x32:ff/ff00/ff0000
-rawfb map:/dev/fb0@1024x768x32
-rawfb map:/dev/fb0@800x480x16#800x960
-rawfb map:/tmp/Xvfb_screen0@640x480x8+3232
Expand Down
1 change: 1 addition & 0 deletions src/Makefile.am
Original file line number Diff line number Diff line change
Expand Up @@ -140,6 +140,7 @@ AM_CPPFLAGS = \
x11vnc_LDADD = \
$(LDADD) \
@SSL_LIBS@ \
@DRM_LIBS@ \
@CRYPT_LIBS@ \
@X_LIBS@ \
@AVAHI_LIBS@ \
Expand Down
123 changes: 121 additions & 2 deletions src/screen.c
Original file line number Diff line number Diff line change
Expand Up @@ -1636,6 +1636,7 @@ void linux_dev_fb_msg(char* q) {
#define RAWFB_MMAP 1
#define RAWFB_FILE 2
#define RAWFB_SHM 3
#define RAWFB_DRM 4

XImage *initialize_raw_fb(int reset) {
char *str, *rstr, *q;
Expand Down Expand Up @@ -1934,6 +1935,7 @@ if (db) fprintf(stderr, "initialize_raw_fb reset\n");
* -rawfb shm:163938442@640x480x32:ff/ff00/ff0000+3000
* -rawfb map:/path/to/file@640x480x32:ff/ff00/ff0000
* -rawfb file:/path/to/file@640x480x32:ff/ff00/ff0000
* -rawfb drm:/dev/dri/card0@640x480x32:ff/ff00/ff0000
*/

if (raw_fb_full_str) {
Expand Down Expand Up @@ -2110,7 +2112,8 @@ if (db) fprintf(stderr, "initialize_raw_fb reset\n");
}

if (strstr(str, "shm:") != str && strstr(str, "mmap:") != str &&
strstr(str, "map:") != str && strstr(str, "file:") != str) {
strstr(str, "map:") != str && strstr(str, "file:") != str &&
strstr(str, "drm:") != str ) {
/* hmmm, not following directions, see if map: applies */
struct stat sbuf;
if (stat(str, &sbuf) == 0) {
Expand All @@ -2130,6 +2133,18 @@ if (db) fprintf(stderr, "initialize_raw_fb reset\n");
if (sscanf(str, "shm:%d", &shmid) == 1) {
/* shm:N */
#if HAVE_XSHM || HAVE_SHMAT
/* try to use shm key*/
key_t shmkey = 0;
int newshmid = -1;

shmkey = shmid;

newshmid = shmget(shmkey, 0, 0);
if (newshmid != -1) {
rfbLog("rawfb: use %d as shm key, shmid is %d \n", shmid, newshmid);
shmid = newshmid;
}

raw_fb_addr = (char *) shmat(shmid, 0, SHM_RDONLY);
if (! raw_fb_addr) {
rfbLogEnable(1);
Expand Down Expand Up @@ -2266,7 +2281,111 @@ if (db) fprintf(stderr, "initialize_raw_fb reset\n");
rfbLog("rawfb: seek file: %s\n", q);
rfbLog(" W: %d H: %d B: %d sz: %d\n", w, h, b, size);
}
} else {
}

else if (strstr(str, "drm:") == str ) {
/* drm:D */
/* drm:/dev/dri/card0 */
#if HAVE_LIBDRM
q = strchr(str, ':');
q++;

/* open DRM device */
int fd = open(q, O_RDWR | O_CLOEXEC);
if (fd < 0) {
rfbLogEnable(1);
rfbLog("can not open drm device: %s\n", str);
clean_up_exit(1);
}

/* get current CRTC */
drmModeRes* res = drmModeGetResources(fd);
if (!res) {
rfbLogEnable(1);
rfbLog("can not read drm device: %s\n", str);
clean_up_exit(1);
}

/* get first crtc */
/* drmModeCrtc* crtc = drmModeGetCrtc(fd, res->crtcs[0]); */
drmModeCrtc* crtc = 0;
drmModeFB* fb = 0;
int dma_buf_fd = 0;
rfbLog("drm crtc count:%d\n", res->count_crtcs);
for (int i = 0; i < res->count_crtcs; i++) {

crtc = drmModeGetCrtc(fd, res->crtcs[i]);
if (crtc->mode_valid && crtc->buffer_id != 0) {

fb = drmModeGetFB(fd, crtc->buffer_id);

struct drm_prime_handle prime_handle = {
.handle = fb->handle,
.fd = -1,
.flags = 0
};

if (ioctl(fd, DRM_IOCTL_PRIME_HANDLE_TO_FD, &prime_handle) < 0) {
drmModeFreeFB(fb);
drmModeFreeCrtc(crtc);
crtc = NULL;
}
else{
dma_buf_fd = prime_handle.fd;
break;

}


}

}

if (!crtc) {
rfbLogEnable(1);
rfbLog("no valid crtc for drm device: %s\n", str);
clean_up_exit(1);
}

if (!dma_buf_fd) {
rfbLogEnable(1);
rfbLog("open CRTC err, please turn on your display. ");
clean_up_exit(1);
}



/* get DMA-BUF size */
off_t size = lseek(dma_buf_fd, 0, SEEK_END);
lseek(dma_buf_fd, 0, SEEK_SET);
rfbLog("dma buffer (%d) size=%ld\n", dma_buf_fd, size);

/* mapping DMA-BUF to memory */
void* map = mmap(NULL, size, PROT_READ, MAP_SHARED, dma_buf_fd, 0);
if (map == MAP_FAILED) {
rfbLogEnable(1);
rfbLog("mmap failed for drm device: %s\n", str);
clean_up_exit(1);
}

/* clean resouces */
/* close(dma_buf_fd); */
drmModeFreeFB(fb);
drmModeFreeCrtc(crtc);
drmModeFreeResources(res);
close(fd);

raw_fb_addr = (char*)map;
last_mode = RAWFB_DRM;
#else
rfbLogEnable(1);
rfbLog("x11vnc was compiled without drm support.\n");
rfbLogPerror("drmModeGetResources");
clean_up_exit(1);
#endif

}
else {
rfbLogEnable(1);
rfbLog("invalid rawfb str: %s\n", str);
clean_up_exit(1);
Expand Down
7 changes: 7 additions & 0 deletions src/x11vnc.h
Original file line number Diff line number Diff line change
Expand Up @@ -69,6 +69,13 @@ so, delete this exception statement from your version.
#include <rfb/rfb.h>
#include <rfb/rfbregion.h>

#if HAVE_LIBDRM
/* drm access */
#include <sys/ioctl.h>
#include <xf86drm.h>
#include <xf86drmMode.h>

#endif

/* we can now build under --without-x: */
#if HAVE_X11
Expand Down

0 comments on commit a61f364

Please sign in to comment.