Skip to content

Commit 1390b33

Browse files
author
Volodymyr Samokhatko
committed
tight: multithreaded
1 parent e4f3e72 commit 1390b33

File tree

5 files changed

+1644
-1310
lines changed

5 files changed

+1644
-1310
lines changed

include/rfb/rfb.h

+16
Original file line numberDiff line numberDiff line change
@@ -84,6 +84,10 @@ typedef UINT32 in_addr_t;
8484
#endif
8585
#endif
8686

87+
/* Maximum number of threads to use for multithreaded encoding, regardless of
88+
the CPU count */
89+
#define MAX_ENCODING_THREADS 8
90+
8791
struct _rfbClientRec;
8892
struct _rfbScreenInfo;
8993
struct rfbCursor;
@@ -372,6 +376,8 @@ typedef struct _rfbScreenInfo
372376
#ifdef LIBVNCSERVER_HAVE_LIBZ
373377
rfbSetXCutTextUTF8ProcPtr setXCutTextUTF8;
374378
#endif
379+
rfbBool rfbMT;
380+
int rfbNumThreads;
375381
} rfbScreenInfo, *rfbScreenInfoPtr;
376382

377383

@@ -705,6 +711,16 @@ typedef struct _rfbClientRec {
705711
rfbBool tightUsePixelFormat24;
706712
void *tightTJ;
707713
int tightPngDstDataLen;
714+
715+
/* Multithreaded tight encoding. */
716+
717+
rfbBool threadInit;
718+
#ifdef LIBVNCSERVER_HAVE_LIBPTHREAD
719+
pthread_t thnd[MAX_ENCODING_THREADS];
720+
#elif defined(LIBVNCSERVER_HAVE_WIN32THREADS)
721+
uintptr_t thnd[MAX_ENCODING_THREADS];
722+
#endif
723+
708724
#endif
709725
#endif
710726
} rfbClientRec, *rfbClientPtr;

src/libvncserver/cargs.c

+21
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,13 @@ rfbUsage(void)
6060
fprintf(stderr, "-listenv6 ipv6addr listen for IPv6 connections only on network interface with\n");
6161
fprintf(stderr, " addr ipv6addr. '-listen localhost' and hostname work too.\n");
6262
#endif
63+
#if defined(LIBVNCSERVER_HAVE_LIBPTHREAD) || defined(LIBVNCSERVER_HAVE_WIN32THREADS)
64+
fprintf(stderr, "-nomt disable multithreaded Tight encoding\n");
65+
fprintf(stderr, "-nthreads N specify number of threads (1 <= N <= %d) to use with\n",
66+
MAX_ENCODING_THREADS);
67+
fprintf(stderr, " multithreaded Tight encoding [default: 1 per CPU core,\n");
68+
fprintf(stderr, " max. %d]\n", MAX_ENCODING_THREADS);
69+
#endif
6370

6471
for(extension=rfbGetExtensionIterator();extension;extension=extension->next)
6572
if(extension->usage)
@@ -202,6 +209,20 @@ rfbProcessArguments(rfbScreenInfoPtr rfbScreen,int* argc, char *argv[])
202209
}
203210
rfbScreen->listen6Interface = argv[++i];
204211
#endif
212+
#if defined(LIBVNCSERVER_HAVE_LIBPTHREAD) || defined(LIBVNCSERVER_HAVE_WIN32THREADS)
213+
} else if (strcmp(argv[i], "-nomt") == 0) {
214+
rfbScreen->rfbMT = FALSE;
215+
} else if (strcmp(argv[i], "-nthreads") == 0) {
216+
if (i + 1 >= *argc) {
217+
rfbUsage();
218+
return FALSE;
219+
}
220+
rfbScreen->rfbNumThreads = atoi(argv[++i]);
221+
if (rfbScreen->rfbNumThreads < 1 || rfbScreen->rfbNumThreads > MAX_ENCODING_THREADS) {
222+
rfbUsage();
223+
return FALSE;
224+
}
225+
#endif
205226
#ifdef LIBVNCSERVER_WITH_WEBSOCKETS
206227
} else if (strcmp(argv[i], "-sslkeyfile") == 0) { /* -sslkeyfile sslkeyfile */
207228
if (i + 1 >= *argc) {

src/libvncserver/main.c

+84
Original file line numberDiff line numberDiff line change
@@ -10,6 +10,13 @@
1010
* see GPL (latest version) for full details
1111
*/
1212

13+
#include <rfb/rfbconfig.h>
14+
15+
#if LIBVNCSERVER_HAVE_SCHED_H
16+
#define _GNU_SOURCE
17+
#include <sched.h>
18+
#endif
19+
1320
#ifdef __STRICT_ANSI__
1421
#define _BSD_SOURCE
1522
#endif
@@ -25,6 +32,10 @@
2532
#define true -1
2633
#endif
2734

35+
#ifdef LIBVNCSERVER_HAVE_SYS_SYSCTL_H
36+
#include <sys/sysctl.h>
37+
#endif
38+
2839
#ifdef LIBVNCSERVER_HAVE_SYS_TIME_H
2940
#include <sys/time.h>
3041
#endif
@@ -55,6 +66,43 @@ char rfbEndianTest = (1==0);
5566
char rfbEndianTest = (1==1);
5667
#endif
5768

69+
#ifndef min
70+
inline static int min(int a, int b)
71+
{
72+
return a > b ? b : a;
73+
}
74+
#endif
75+
76+
#ifdef __APPLE__
77+
typedef struct cpu_set {
78+
uint64_t count;
79+
} cpu_set_t;
80+
81+
static inline void
82+
CPU_ZERO(cpu_set_t *cs) { cs->count = 0; }
83+
84+
static inline int
85+
CPU_COUNT(cpu_set_t *cs) { return __builtin_popcountll(cs->count); }
86+
87+
int sched_getaffinity(pid_t pid, size_t cpu_size, cpu_set_t *cpu_set)
88+
{
89+
int i;
90+
int32_t core_count = 0;
91+
size_t len = sizeof(core_count);
92+
int ret = sysctlbyname("machdep.cpu.core_count", &core_count, &len, 0, 0);
93+
if (ret) {
94+
rfbErr("error while get core count %d\n", ret);
95+
return -1;
96+
}
97+
cpu_set->count = 0;
98+
for (i = 0; i < core_count; i++) {
99+
cpu_set->count |= (1 << i);
100+
}
101+
102+
return 0;
103+
}
104+
#endif
105+
58106
/*
59107
* Protocol extensions
60108
*/
@@ -996,11 +1044,47 @@ rfbScreenInfoPtr rfbGetScreen(int* argc,char** argv,
9961044

9971045
screen->permitFileTransfer = FALSE;
9981046

1047+
screen->rfbMT = TRUE;
1048+
screen->rfbNumThreads = 0;
1049+
9991050
if(!rfbProcessArguments(screen,argc,argv)) {
10001051
free(screen);
10011052
return NULL;
10021053
}
10031054

1055+
#if defined(LIBVNCSERVER_HAVE_LIBPTHREAD) || defined(LIBVNCSERVER_HAVE_WIN32THREADS)
1056+
#if defined(WIN32) || defined(__MINGW32__)
1057+
DWORD64 dwProcessAffinity, dwSystemAffinity;
1058+
GetProcessAffinityMask(GetCurrentProcess(), &dwProcessAffinity, &dwSystemAffinity);
1059+
#if defined(__MINGW32__)
1060+
const int np = __builtin_popcountll(dwProcessAffinity);
1061+
#else
1062+
const int np = __popcnt64(dwProcessAffinity);
1063+
#endif
1064+
#elif LIBVNCSERVER_HAVE_SCHED_H
1065+
cpu_set_t cs;
1066+
CPU_ZERO(&cs);
1067+
int cpus = -1;
1068+
if (sched_getaffinity(0, sizeof(cs), &cs) == 0) {
1069+
cpus = CPU_COUNT(&cs);
1070+
}
1071+
const int np = cpus;
1072+
#else
1073+
const int np = -1;
1074+
#endif
1075+
if (np == -1 && screen->rfbMT) {
1076+
rfbLog("WARNING: Could not determine CPU count. Multithreaded encoding disabled.\n");
1077+
screen->rfbMT = FALSE;
1078+
}
1079+
if (!screen->rfbMT) screen->rfbNumThreads = 1;
1080+
else if (screen->rfbNumThreads < 1) screen->rfbNumThreads = min(np, 4);
1081+
if (screen->rfbNumThreads > np) {
1082+
rfbLog("NOTICE: Encoding thread count has been clamped to CPU count\n");
1083+
screen->rfbNumThreads = np;
1084+
}
1085+
#endif
1086+
1087+
10041088
#ifdef WIN32
10051089
{
10061090
DWORD dummy=255;

src/libvncserver/rfbserver.c

+9
Original file line numberDiff line numberDiff line change
@@ -504,6 +504,15 @@ rfbNewTCPOrUDPClient(rfbScreenInfoPtr rfbScreen,
504504
}
505505
}
506506

507+
#ifdef LIBVNCSERVER_HAVE_LIBJPEG
508+
/* Multithreaded tight encoding. */
509+
510+
cl->threadInit = FALSE;
511+
#if defined(LIBVNCSERVER_HAVE_LIBPTHREAD) || defined(LIBVNCSERVER_HAVE_WIN32THREADS)
512+
memset(cl->thnd, 0, sizeof(cl->thnd));
513+
#endif
514+
#endif
515+
507516
for(extension = rfbGetExtensionIterator(); extension;
508517
extension=extension->next) {
509518
void* data = NULL;

0 commit comments

Comments
 (0)