mirror of
https://github.com/kasmtech/KasmVNC.git
synced 2025-01-07 14:39:48 +01:00
dc21d5f97c
Co-authored-by: Lauri Kasanen <cand@gmx.com>
632 lines
15 KiB
C++
632 lines
15 KiB
C++
/* Copyright (C) 2021 Kasm
|
|
*
|
|
* This is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* This software is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this software; if not, write to the Free Software
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307,
|
|
* USA.
|
|
*/
|
|
|
|
#define __STDC_FORMAT_MACROS
|
|
|
|
#include <inttypes.h>
|
|
#include <network/GetAPI.h>
|
|
#include <rfb/ConnParams.h>
|
|
#include <rfb/EncodeManager.h>
|
|
#include <rfb/LogWriter.h>
|
|
#include <rfb/JpegCompressor.h>
|
|
#include <rfb/xxhash.h>
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
using namespace network;
|
|
using namespace rfb;
|
|
|
|
static LogWriter vlog("GetAPIMessager");
|
|
|
|
struct TightJPEGConfiguration {
|
|
int quality;
|
|
int subsampling;
|
|
};
|
|
|
|
static const struct TightJPEGConfiguration conf[10] = {
|
|
{ 15, subsample4X }, // 0
|
|
{ 29, subsample4X }, // 1
|
|
{ 41, subsample4X }, // 2
|
|
{ 42, subsample2X }, // 3
|
|
{ 62, subsample2X }, // 4
|
|
{ 77, subsample2X }, // 5
|
|
{ 79, subsampleNone }, // 6
|
|
{ 86, subsampleNone }, // 7
|
|
{ 92, subsampleNone }, // 8
|
|
{ 100, subsampleNone } // 9
|
|
};
|
|
|
|
GetAPIMessager::GetAPIMessager(const char *passwdfile_): passwdfile(passwdfile_),
|
|
screenW(0), screenH(0), screenHash(0),
|
|
cachedW(0), cachedH(0), cachedQ(0),
|
|
ownerConnected(0), activeUsers(0) {
|
|
|
|
pthread_mutex_init(&screenMutex, NULL);
|
|
pthread_mutex_init(&userMutex, NULL);
|
|
pthread_mutex_init(&statMutex, NULL);
|
|
pthread_mutex_init(&frameStatMutex, NULL);
|
|
pthread_mutex_init(&userInfoMutex, NULL);
|
|
|
|
serverFrameStats.inprogress = 0;
|
|
}
|
|
|
|
// from main thread
|
|
void GetAPIMessager::mainUpdateScreen(rfb::PixelBuffer *pb) {
|
|
if (pthread_mutex_trylock(&screenMutex))
|
|
return;
|
|
|
|
int stride;
|
|
const rdr::U8 * const buf = pb->getBuffer(pb->getRect(), &stride);
|
|
|
|
if (pb->width() != screenW || pb->height() != screenH) {
|
|
screenHash = 0;
|
|
screenW = pb->width();
|
|
screenH = pb->height();
|
|
screenPb.setPF(pb->getPF());
|
|
screenPb.setSize(screenW, screenH);
|
|
|
|
cachedW = cachedH = cachedQ = 0;
|
|
cachedJpeg.clear();
|
|
}
|
|
|
|
const uint64_t newHash = XXH64(buf, pb->area() * 4, 0);
|
|
if (newHash != screenHash) {
|
|
cachedW = cachedH = cachedQ = 0;
|
|
cachedJpeg.clear();
|
|
|
|
screenHash = newHash;
|
|
rdr::U8 *rw = screenPb.getBufferRW(screenPb.getRect(), &stride);
|
|
memcpy(rw, buf, screenW * screenH * 4);
|
|
screenPb.commitBufferRW(screenPb.getRect());
|
|
}
|
|
|
|
pthread_mutex_unlock(&screenMutex);
|
|
}
|
|
|
|
void GetAPIMessager::mainUpdateBottleneckStats(const char userid[], const char stats[]) {
|
|
if (pthread_mutex_trylock(&statMutex))
|
|
return;
|
|
|
|
bottleneckStats[userid] = stats;
|
|
|
|
pthread_mutex_unlock(&statMutex);
|
|
}
|
|
|
|
void GetAPIMessager::mainClearBottleneckStats(const char userid[]) {
|
|
if (pthread_mutex_lock(&statMutex))
|
|
return;
|
|
|
|
bottleneckStats.erase(userid);
|
|
|
|
pthread_mutex_unlock(&statMutex);
|
|
}
|
|
|
|
void GetAPIMessager::mainUpdateServerFrameStats(uint8_t changedPerc,
|
|
uint32_t all, uint32_t jpeg, uint32_t webp, uint32_t analysis,
|
|
uint32_t jpegarea, uint32_t webparea,
|
|
uint16_t njpeg, uint16_t nwebp,
|
|
uint16_t enc, uint16_t scale, uint16_t shot,
|
|
uint16_t w, uint16_t h) {
|
|
|
|
if (pthread_mutex_lock(&frameStatMutex))
|
|
return;
|
|
|
|
serverFrameStats.changedPerc = changedPerc;
|
|
serverFrameStats.all = all;
|
|
serverFrameStats.jpeg = jpeg;
|
|
serverFrameStats.webp = webp;
|
|
serverFrameStats.analysis = analysis;
|
|
serverFrameStats.jpegarea = jpegarea;
|
|
serverFrameStats.webparea = webparea;
|
|
serverFrameStats.njpeg = njpeg;
|
|
serverFrameStats.nwebp = nwebp;
|
|
serverFrameStats.enc = enc;
|
|
serverFrameStats.scale = scale;
|
|
serverFrameStats.shot = shot;
|
|
serverFrameStats.w = w;
|
|
serverFrameStats.h = h;
|
|
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
}
|
|
|
|
void GetAPIMessager::mainUpdateClientFrameStats(const char userid[], uint32_t render,
|
|
uint32_t all, uint32_t ping) {
|
|
|
|
if (pthread_mutex_lock(&frameStatMutex))
|
|
return;
|
|
|
|
clientFrameStats_t s;
|
|
s.render = render;
|
|
s.all = all;
|
|
s.ping = ping;
|
|
|
|
clientFrameStats[userid] = s;
|
|
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
}
|
|
|
|
void GetAPIMessager::mainUpdateUserInfo(const uint8_t ownerConn, const uint8_t numUsers) {
|
|
if (pthread_mutex_lock(&userInfoMutex))
|
|
return;
|
|
|
|
ownerConnected = ownerConn;
|
|
activeUsers = numUsers;
|
|
|
|
pthread_mutex_unlock(&userInfoMutex);
|
|
}
|
|
|
|
// from network threads
|
|
uint8_t *GetAPIMessager::netGetScreenshot(uint16_t w, uint16_t h,
|
|
const uint8_t q, const bool dedup,
|
|
uint32_t &len, uint8_t *staging) {
|
|
|
|
uint8_t *ret = NULL;
|
|
len = 0;
|
|
|
|
if (w > screenW)
|
|
w = screenW;
|
|
if (h > screenH)
|
|
h = screenH;
|
|
|
|
if (!screenW || !screenH)
|
|
vlog.error("Screenshot requested but no screenshot exists (screen hasn't been viewed)");
|
|
|
|
if (!w || !h || q > 9 || !staging)
|
|
return NULL;
|
|
|
|
if (pthread_mutex_lock(&screenMutex))
|
|
return NULL;
|
|
|
|
if (w == cachedW && h == cachedH && q == cachedQ) {
|
|
if (dedup) {
|
|
// Return the hash of the unchanged image
|
|
sprintf((char *) staging, "%" PRIx64, screenHash);
|
|
ret = staging;
|
|
len = 16;
|
|
} else {
|
|
// Return the cached image
|
|
len = cachedJpeg.size();
|
|
ret = staging;
|
|
memcpy(ret, &cachedJpeg[0], len);
|
|
|
|
vlog.info("Returning cached screenshot");
|
|
}
|
|
} else {
|
|
// Encode the new JPEG, cache it
|
|
JpegCompressor jc;
|
|
int quality, subsampling;
|
|
|
|
quality = conf[q].quality;
|
|
subsampling = conf[q].subsampling;
|
|
|
|
jc.clear();
|
|
int stride;
|
|
|
|
if (w != screenW || h != screenH) {
|
|
float xdiff = w / (float) screenW;
|
|
float ydiff = h / (float) screenH;
|
|
const float diff = xdiff < ydiff ? xdiff : ydiff;
|
|
|
|
const uint16_t neww = screenW * diff;
|
|
const uint16_t newh = screenH * diff;
|
|
|
|
const PixelBuffer *scaled = progressiveBilinearScale(&screenPb, neww, newh, diff);
|
|
const rdr::U8 * const buf = scaled->getBuffer(scaled->getRect(), &stride);
|
|
|
|
jc.compress(buf, stride, scaled->getRect(),
|
|
scaled->getPF(), quality, subsampling);
|
|
|
|
cachedJpeg.resize(jc.length());
|
|
memcpy(&cachedJpeg[0], jc.data(), jc.length());
|
|
|
|
delete scaled;
|
|
|
|
vlog.info("Returning scaled screenshot");
|
|
} else {
|
|
const rdr::U8 * const buf = screenPb.getBuffer(screenPb.getRect(), &stride);
|
|
|
|
jc.compress(buf, stride, screenPb.getRect(),
|
|
screenPb.getPF(), quality, subsampling);
|
|
|
|
cachedJpeg.resize(jc.length());
|
|
memcpy(&cachedJpeg[0], jc.data(), jc.length());
|
|
|
|
vlog.info("Returning normal screenshot");
|
|
}
|
|
|
|
cachedQ = q;
|
|
cachedW = w;
|
|
cachedH = h;
|
|
|
|
len = cachedJpeg.size();
|
|
ret = staging;
|
|
memcpy(ret, &cachedJpeg[0], len);
|
|
}
|
|
|
|
pthread_mutex_unlock(&screenMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
#define USERNAME_LEN sizeof(((struct kasmpasswd_entry_t *)0)->user)
|
|
#define PASSWORD_LEN sizeof(((struct kasmpasswd_entry_t *)0)->password)
|
|
|
|
uint8_t GetAPIMessager::netAddUser(const char name[], const char pw[], const bool write) {
|
|
if (strlen(name) >= USERNAME_LEN) {
|
|
vlog.error("Username too long");
|
|
return 0;
|
|
}
|
|
|
|
if (strlen(pw) >= PASSWORD_LEN) {
|
|
vlog.error("Password too long");
|
|
return 0;
|
|
}
|
|
|
|
if (!passwdfile)
|
|
return 0;
|
|
|
|
action_data act;
|
|
|
|
memcpy(act.data.user, name, USERNAME_LEN);
|
|
act.data.user[USERNAME_LEN - 1] = '\0';
|
|
memcpy(act.data.password, pw, PASSWORD_LEN);
|
|
act.data.password[PASSWORD_LEN - 1] = '\0';
|
|
act.data.owner = 0;
|
|
act.data.write = write;
|
|
|
|
if (pthread_mutex_lock(&userMutex))
|
|
return 0;
|
|
|
|
// This needs to be handled locally for proper interactivity
|
|
// (consider adding users when nobody is connected).
|
|
// The mutex and atomic rename keep things in sync.
|
|
|
|
struct kasmpasswd_t *set = readkasmpasswd(passwdfile);
|
|
unsigned s;
|
|
for (s = 0; s < set->num; s++) {
|
|
if (!strcmp(set->entries[s].user, act.data.user)) {
|
|
vlog.error("Can't create user %s, already exists", act.data.user);
|
|
goto out;
|
|
}
|
|
}
|
|
|
|
s = set->num++;
|
|
set->entries = (struct kasmpasswd_entry_t *) realloc(set->entries,
|
|
set->num * sizeof(struct kasmpasswd_entry_t));
|
|
set->entries[s] = act.data;
|
|
|
|
writekasmpasswd(passwdfile, set);
|
|
vlog.info("User %s created", act.data.user);
|
|
out:
|
|
pthread_mutex_unlock(&userMutex);
|
|
|
|
return 1;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netRemoveUser(const char name[]) {
|
|
if (strlen(name) >= USERNAME_LEN) {
|
|
vlog.error("Username too long");
|
|
return 0;
|
|
}
|
|
|
|
action_data act;
|
|
act.action = USER_REMOVE;
|
|
|
|
memcpy(act.data.user, name, USERNAME_LEN);
|
|
act.data.user[USERNAME_LEN - 1] = '\0';
|
|
|
|
if (pthread_mutex_lock(&userMutex))
|
|
return 0;
|
|
|
|
actionQueue.push_back(act);
|
|
|
|
pthread_mutex_unlock(&userMutex);
|
|
|
|
return 1;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netGiveControlTo(const char name[]) {
|
|
if (strlen(name) >= USERNAME_LEN) {
|
|
vlog.error("Username too long");
|
|
return 0;
|
|
}
|
|
|
|
action_data act;
|
|
act.action = USER_GIVE_CONTROL;
|
|
|
|
memcpy(act.data.user, name, USERNAME_LEN);
|
|
act.data.user[USERNAME_LEN - 1] = '\0';
|
|
|
|
if (pthread_mutex_lock(&userMutex))
|
|
return 0;
|
|
|
|
actionQueue.push_back(act);
|
|
|
|
pthread_mutex_unlock(&userMutex);
|
|
|
|
return 1;
|
|
}
|
|
|
|
void GetAPIMessager::netGetBottleneckStats(char *buf, uint32_t len) {
|
|
/*
|
|
{
|
|
"username.1": {
|
|
"192.168.100.2:14908": [ 100, 100, 100, 100 ],
|
|
"192.168.100.3:14918": [ 100, 100, 100, 100 ]
|
|
},
|
|
"username.2": {
|
|
"192.168.100.5:14904": [ 100, 100, 100, 100 ]
|
|
}
|
|
}
|
|
*/
|
|
std::map<std::string, std::string>::const_iterator it;
|
|
const char *prev = NULL;
|
|
FILE *f;
|
|
|
|
if (pthread_mutex_lock(&statMutex)) {
|
|
buf[0] = 0;
|
|
return;
|
|
}
|
|
|
|
// Conservative estimate
|
|
if (len < bottleneckStats.size() * 60) {
|
|
buf[0] = 0;
|
|
goto out;
|
|
}
|
|
|
|
f = fmemopen(buf, len, "w");
|
|
|
|
fprintf(f, "{\n");
|
|
|
|
for (it = bottleneckStats.begin(); it != bottleneckStats.end(); it++) {
|
|
// user@127.0.0.1_1627311208.791752::websocket
|
|
const char *id = it->first.c_str();
|
|
const char *data = it->second.c_str();
|
|
|
|
const char *at = strchr(id, '@');
|
|
if (!at)
|
|
continue;
|
|
|
|
const unsigned userlen = at - id;
|
|
if (prev && !strncmp(prev, id, userlen)) {
|
|
// Same user
|
|
fprintf(f, ",\n\t\t\"%s\": %s", at + 1, data);
|
|
} else {
|
|
// New one
|
|
if (prev) {
|
|
fprintf(f, "\n\t},\n");
|
|
}
|
|
fprintf(f, "\t\"%.*s\": {\n", userlen, id);
|
|
fprintf(f, "\t\t\"%s\": %s", at + 1, data);
|
|
}
|
|
|
|
prev = id;
|
|
}
|
|
|
|
if (!bottleneckStats.size())
|
|
fprintf(f, "}\n");
|
|
else
|
|
fprintf(f, "\n\t}\n}\n");
|
|
|
|
fclose(f);
|
|
|
|
out:
|
|
pthread_mutex_unlock(&statMutex);
|
|
}
|
|
|
|
void GetAPIMessager::netGetFrameStats(char *buf, uint32_t len) {
|
|
/*
|
|
{
|
|
"frame" : {
|
|
"resx": 1024,
|
|
"resy": 1280,
|
|
"changed": 75,
|
|
"server_time": 23
|
|
},
|
|
"server_side" : [
|
|
{ "process_name": "Analysis", "time": 20 },
|
|
{ "process_name": "TightWEBPEncoder", "time": 20, "count": 64, "area": 12 },
|
|
{ "process_name": "TightJPEGEncoder", "time": 20, "count": 64, "area": 12 }
|
|
],
|
|
"client_side" : [
|
|
{
|
|
"client": "123.1.2.1:1211",
|
|
"client_time": 20,
|
|
"ping": 20,
|
|
"processes" : [
|
|
{ "process_name": "scanRenderQ", "time": 20 }
|
|
]
|
|
}
|
|
}
|
|
}
|
|
*/
|
|
std::map<std::string, clientFrameStats_t>::const_iterator it;
|
|
unsigned i = 0;
|
|
FILE *f;
|
|
|
|
if (pthread_mutex_lock(&frameStatMutex)) {
|
|
buf[0] = 0;
|
|
return;
|
|
}
|
|
|
|
const unsigned num = clientFrameStats.size();
|
|
|
|
// Conservative estimate
|
|
if (len < 1024) {
|
|
buf[0] = 0;
|
|
goto out;
|
|
}
|
|
|
|
f = fmemopen(buf, len, "w");
|
|
|
|
fprintf(f, "{\n");
|
|
|
|
fprintf(f, "\t\"frame\" : {\n"
|
|
"\t\t\"resx\": %u,\n"
|
|
"\t\t\"resy\": %u,\n"
|
|
"\t\t\"changed\": %u,\n"
|
|
"\t\t\"server_time\": %u\n"
|
|
"\t},\n",
|
|
serverFrameStats.w,
|
|
serverFrameStats.h,
|
|
serverFrameStats.changedPerc,
|
|
serverFrameStats.all);
|
|
|
|
fprintf(f, "\t\"server_side\" : [\n"
|
|
"\t\t{ \"process_name\": \"Analysis\", \"time\": %u },\n"
|
|
"\t\t{ \"process_name\": \"Screenshot\", \"time\": %u },\n"
|
|
"\t\t{ \"process_name\": \"Encoding_total\", \"time\": %u, \"videoscaling\": %u },\n"
|
|
"\t\t{ \"process_name\": \"TightJPEGEncoder\", \"time\": %u, \"count\": %u, \"area\": %u },\n"
|
|
"\t\t{ \"process_name\": \"TightWEBPEncoder\", \"time\": %u, \"count\": %u, \"area\": %u }\n"
|
|
"\t],\n",
|
|
serverFrameStats.analysis,
|
|
serverFrameStats.shot,
|
|
serverFrameStats.enc,
|
|
serverFrameStats.scale,
|
|
serverFrameStats.jpeg,
|
|
serverFrameStats.njpeg,
|
|
serverFrameStats.jpegarea,
|
|
serverFrameStats.webp,
|
|
serverFrameStats.nwebp,
|
|
serverFrameStats.webparea);
|
|
|
|
fprintf(f, "\t\"client_side\" : [\n");
|
|
|
|
for (it = clientFrameStats.begin(); it != clientFrameStats.end(); it++, i++) {
|
|
const char *id = it->first.c_str();
|
|
const clientFrameStats_t &s = it->second;
|
|
|
|
fprintf(f, "\t\t\{\n"
|
|
"\t\t\t\"client\": \"%s\",\n"
|
|
"\t\t\t\"client_time\": %u,\n"
|
|
"\t\t\t\"ping\": %u,\n"
|
|
"\t\t\t\"processes\" : [\n"
|
|
"\t\t\t\t{ \"process_name\": \"scanRenderQ\", \"time\": %u }\n"
|
|
"\t\t\t]\n"
|
|
"\t\t}",
|
|
id,
|
|
s.all,
|
|
s.ping,
|
|
s.render);
|
|
|
|
if (i == num - 1)
|
|
fprintf(f, "\n");
|
|
else
|
|
fprintf(f, ",\n");
|
|
}
|
|
|
|
fprintf(f, "\t]\n}\n");
|
|
|
|
fclose(f);
|
|
|
|
serverFrameStats.inprogress = 0;
|
|
|
|
out:
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netRequestFrameStats(USER_ACTION what, const char *client) {
|
|
// Return 1 for success
|
|
action_data act;
|
|
act.action = what;
|
|
if (client) {
|
|
strncpy(act.data.password, client, PASSWORD_LEN);
|
|
act.data.password[PASSWORD_LEN - 1] = '\0';
|
|
}
|
|
|
|
// In progress already?
|
|
bool fail = false;
|
|
if (pthread_mutex_lock(&frameStatMutex))
|
|
return 0;
|
|
|
|
if (serverFrameStats.inprogress) {
|
|
fail = true;
|
|
vlog.error("Frame stats request already in progress, refusing another");
|
|
} else {
|
|
clientFrameStats.clear();
|
|
memset(&serverFrameStats, 0, sizeof(serverFrameStats_t));
|
|
serverFrameStats.inprogress = 1;
|
|
}
|
|
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
if (fail)
|
|
return 0;
|
|
|
|
// Send it in
|
|
if (pthread_mutex_lock(&userMutex))
|
|
return 0;
|
|
|
|
actionQueue.push_back(act);
|
|
|
|
pthread_mutex_unlock(&userMutex);
|
|
|
|
return 1;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netOwnerConnected() {
|
|
uint8_t ret;
|
|
|
|
if (pthread_mutex_lock(&userInfoMutex))
|
|
return 0;
|
|
|
|
ret = ownerConnected;
|
|
|
|
pthread_mutex_unlock(&userInfoMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netNumActiveUsers() {
|
|
uint8_t ret;
|
|
|
|
if (pthread_mutex_lock(&userInfoMutex))
|
|
return 0;
|
|
|
|
ret = activeUsers;
|
|
|
|
pthread_mutex_unlock(&userInfoMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netGetClientFrameStatsNum() {
|
|
uint8_t ret;
|
|
|
|
if (pthread_mutex_lock(&frameStatMutex))
|
|
return 0;
|
|
|
|
ret = clientFrameStats.size();
|
|
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
|
|
return ret;
|
|
}
|
|
|
|
uint8_t GetAPIMessager::netServerFrameStatsReady() {
|
|
uint8_t ret;
|
|
|
|
if (pthread_mutex_lock(&frameStatMutex))
|
|
return 0;
|
|
|
|
ret = serverFrameStats.w != 0;
|
|
|
|
pthread_mutex_unlock(&frameStatMutex);
|
|
|
|
return ret;
|
|
}
|