2020-04-14 05:16:01 +08:00
|
|
|
#include <dlfcn.h>
|
2020-03-26 14:21:34 +08:00
|
|
|
#include "zm.h"
|
|
|
|
#include "zm_signal.h"
|
|
|
|
#include "zm_libvnc_camera.h"
|
2020-03-29 06:17:19 +08:00
|
|
|
#include "zm_swscale.h"
|
2020-03-26 14:21:34 +08:00
|
|
|
|
2020-03-26 15:08:00 +08:00
|
|
|
#if HAVE_LIBVNC
|
|
|
|
|
2020-03-26 14:21:34 +08:00
|
|
|
static int TAG_0;
|
|
|
|
static int TAG_1;
|
|
|
|
static int TAG_2;
|
2020-04-14 05:16:01 +08:00
|
|
|
static void *libvnc_lib = nullptr;
|
|
|
|
static void *(*rfbClientGetClientData_f)(rfbClient*, void*) = nullptr;
|
|
|
|
static rfbClient *(*rfbGetClient_f)(int, int, int) = nullptr;
|
|
|
|
static void (*rfbClientSetClientData_f)(rfbClient*, void*, void*) = nullptr;
|
|
|
|
static rfbBool (*rfbInitClient_f)(rfbClient*, int*, char**) = nullptr;
|
|
|
|
static void (*rfbClientCleanup_f)(rfbClient*) = nullptr;
|
|
|
|
static int (*WaitForMessage_f)(rfbClient*, unsigned int) = nullptr;
|
|
|
|
static rfbBool (*HandleRFBServerMessage_f)(rfbClient*) = nullptr;
|
|
|
|
|
|
|
|
void bind_libvnc_symbols() {
|
2020-09-09 22:25:55 +08:00
|
|
|
if ( libvnc_lib != nullptr ) // Safe-check
|
2020-04-14 05:16:01 +08:00
|
|
|
return;
|
|
|
|
|
|
|
|
libvnc_lib = dlopen("libvncclient.so", RTLD_LAZY | RTLD_GLOBAL);
|
2020-09-09 22:25:55 +08:00
|
|
|
if ( !libvnc_lib ) {
|
|
|
|
Error("Error loading libvncclient: %s", dlerror());
|
2020-04-14 05:16:01 +08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
*(void**) (&rfbClientGetClientData_f) = dlsym(libvnc_lib, "rfbClientGetClientData");
|
|
|
|
*(void**) (&rfbGetClient_f) = dlsym(libvnc_lib, "rfbGetClient");
|
|
|
|
*(void**) (&rfbClientSetClientData_f) = dlsym(libvnc_lib, "rfbClientSetClientData");
|
|
|
|
*(void**) (&rfbInitClient_f) = dlsym(libvnc_lib, "rfbInitClient");
|
|
|
|
*(void**) (&rfbClientCleanup_f) = dlsym(libvnc_lib, "rfbClientCleanup");
|
|
|
|
*(void**) (&WaitForMessage_f) = dlsym(libvnc_lib, "WaitForMessage");
|
|
|
|
*(void**) (&HandleRFBServerMessage_f) = dlsym(libvnc_lib, "HandleRFBServerMessage");
|
|
|
|
}
|
2020-03-26 14:21:34 +08:00
|
|
|
|
|
|
|
static void GotFrameBufferUpdateCallback(rfbClient *rfb, int x, int y, int w, int h){
|
2020-04-14 05:16:01 +08:00
|
|
|
VncPrivateData *data = (VncPrivateData *)(*rfbClientGetClientData_f)(rfb, &TAG_0);
|
2020-03-26 14:21:34 +08:00
|
|
|
data->buffer = rfb->frameBuffer;
|
2020-11-18 00:08:11 +08:00
|
|
|
Debug(1, "GotFrameBufferUpdateallback x:%d y:%d w%d h:%d width: %d, height: %d",
|
|
|
|
x,y,w,h, rfb->width, rfb->height);
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static char* GetPasswordCallback(rfbClient* cl){
|
2020-11-18 00:08:11 +08:00
|
|
|
Debug(1, "Getcredentials: %s", (*rfbClientGetClientData_f)(cl, &TAG_1));
|
|
|
|
return strdup(
|
|
|
|
(const char *)(*rfbClientGetClientData_f)(cl, &TAG_1));
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
static rfbCredential* GetCredentialsCallback(rfbClient* cl, int credentialType){
|
|
|
|
rfbCredential *c = (rfbCredential *)malloc(sizeof(rfbCredential));
|
2020-03-29 04:03:52 +08:00
|
|
|
if ( credentialType != rfbCredentialTypeUser ) {
|
2020-09-09 22:25:55 +08:00
|
|
|
free(c);
|
|
|
|
return nullptr;
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
2020-11-18 00:08:11 +08:00
|
|
|
Debug(1, "Getcredentials: %s:%s", (*rfbClientGetClientData_f)(cl, &TAG_1), (*rfbClientGetClientData_f)(cl, &TAG_2));
|
2020-04-14 05:16:01 +08:00
|
|
|
c->userCredential.password = strdup((const char *)(*rfbClientGetClientData_f)(cl, &TAG_1));
|
|
|
|
c->userCredential.username = strdup((const char *)(*rfbClientGetClientData_f)(cl, &TAG_2));
|
2020-03-26 14:21:34 +08:00
|
|
|
return c;
|
|
|
|
}
|
|
|
|
|
|
|
|
VncCamera::VncCamera(
|
|
|
|
unsigned int p_monitor_id,
|
|
|
|
const std::string &host,
|
|
|
|
const std::string &port,
|
|
|
|
const std::string &user,
|
|
|
|
const std::string &pass,
|
|
|
|
int p_width,
|
|
|
|
int p_height,
|
|
|
|
int p_colours,
|
|
|
|
int p_brightness,
|
|
|
|
int p_contrast,
|
|
|
|
int p_hue,
|
|
|
|
int p_colour,
|
|
|
|
bool p_capture,
|
|
|
|
bool p_record_audio ) :
|
|
|
|
Camera(
|
|
|
|
p_monitor_id,
|
|
|
|
VNC_SRC,
|
|
|
|
p_width,
|
|
|
|
p_height,
|
|
|
|
p_colours,
|
|
|
|
ZM_SUBPIX_ORDER_DEFAULT_FOR_COLOUR(p_colours),
|
|
|
|
p_brightness,
|
|
|
|
p_contrast,
|
|
|
|
p_hue,
|
|
|
|
p_colour,
|
|
|
|
p_capture,
|
|
|
|
p_record_audio
|
|
|
|
),
|
2020-11-18 00:08:11 +08:00
|
|
|
mRfb(nullptr),
|
2020-03-26 14:21:34 +08:00
|
|
|
mHost(host),
|
|
|
|
mPort(port),
|
|
|
|
mUser(user),
|
|
|
|
mPass(pass)
|
|
|
|
{
|
|
|
|
Debug(2, "Host:%s Port: %s User: %s Pass:%s", mHost.c_str(), mPort.c_str(), mUser.c_str(), mPass.c_str());
|
2020-03-29 06:17:19 +08:00
|
|
|
|
|
|
|
if ( colours == ZM_COLOUR_RGB32 ) {
|
|
|
|
subpixelorder = ZM_SUBPIX_ORDER_RGBA;
|
|
|
|
mImgPixFmt = AV_PIX_FMT_RGBA;
|
|
|
|
mBpp = 4;
|
|
|
|
} else if ( colours == ZM_COLOUR_RGB24 ) {
|
|
|
|
subpixelorder = ZM_SUBPIX_ORDER_RGB;
|
|
|
|
mImgPixFmt = AV_PIX_FMT_RGB24;
|
|
|
|
mBpp = 3;
|
|
|
|
} else if ( colours == ZM_COLOUR_GRAY8 ) {
|
|
|
|
subpixelorder = ZM_SUBPIX_ORDER_NONE;
|
|
|
|
mImgPixFmt = AV_PIX_FMT_GRAY8;
|
|
|
|
mBpp = 1;
|
|
|
|
} else {
|
|
|
|
Panic("Unexpected colours: %d", colours);
|
|
|
|
}
|
|
|
|
|
2020-11-18 00:08:11 +08:00
|
|
|
if ( capture ) {
|
|
|
|
Debug(3, "Initializing Client");
|
|
|
|
bind_libvnc_symbols();
|
|
|
|
scale.init();
|
2020-03-26 14:21:34 +08:00
|
|
|
|
2020-11-18 00:08:11 +08:00
|
|
|
}
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
2020-11-18 00:08:11 +08:00
|
|
|
VncCamera::~VncCamera() {
|
|
|
|
if ( capture ) {
|
|
|
|
if ( mRfb->frameBuffer )
|
|
|
|
free(mRfb->frameBuffer);
|
|
|
|
(*rfbClientCleanup_f)(mRfb);
|
|
|
|
}
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int VncCamera::PrimeCapture() {
|
2020-07-14 03:03:04 +08:00
|
|
|
Debug(1, "Priming capture from %s", mHost.c_str());
|
2020-11-18 00:08:11 +08:00
|
|
|
|
|
|
|
if ( ! mRfb ) {
|
|
|
|
mVncData.buffer = nullptr;
|
|
|
|
mVncData.width = 0;
|
|
|
|
mVncData.height = 0;
|
|
|
|
|
|
|
|
mRfb = (*rfbGetClient_f)(8 /* bits per sample */, 3 /* samples per pixel */, 4 /* bytes Per Pixel */);
|
|
|
|
mRfb->frameBuffer = (uint8_t *)av_malloc(8*4*width*height);
|
|
|
|
mRfb->canHandleNewFBSize = false;
|
|
|
|
|
|
|
|
(*rfbClientSetClientData_f)(mRfb, &TAG_0, &mVncData);
|
|
|
|
(*rfbClientSetClientData_f)(mRfb, &TAG_1, (void *)mPass.c_str());
|
|
|
|
(*rfbClientSetClientData_f)(mRfb, &TAG_2, (void *)mUser.c_str());
|
|
|
|
|
|
|
|
mRfb->GotFrameBufferUpdate = GotFrameBufferUpdateCallback;
|
|
|
|
mRfb->GetPassword = GetPasswordCallback;
|
|
|
|
mRfb->GetCredential = GetCredentialsCallback;
|
|
|
|
|
|
|
|
mRfb->programName = "Zoneminder VNC Monitor";
|
|
|
|
mRfb->serverHost = strdup(mHost.c_str());
|
|
|
|
mRfb->serverPort = atoi(mPort.c_str());
|
|
|
|
}
|
2020-09-10 00:55:08 +08:00
|
|
|
if ( ! (*rfbInitClient_f)(mRfb, 0, nullptr) ) {
|
2020-11-18 00:08:11 +08:00
|
|
|
/* IF rfbInitClient fails, it calls rdbClientCleanup which will free mRfb */
|
|
|
|
Warning("Failed to Priming capture from %s", mHost.c_str());
|
|
|
|
mRfb = nullptr;
|
2020-09-10 00:55:08 +08:00
|
|
|
return -1;
|
|
|
|
}
|
2020-12-24 07:18:30 +08:00
|
|
|
if ( (mRfb->width != (unsigned int)width) or (mRfb->height != (unsigned int)height) ) {
|
2020-11-18 00:08:11 +08:00
|
|
|
Warning("Specified dimensions do not match screen size monitor: (%dx%d) != vnc: (%dx%d)",
|
|
|
|
width, height, mRfb->width, mRfb->height);
|
|
|
|
}
|
|
|
|
|
2020-03-26 14:21:34 +08:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int VncCamera::PreCapture() {
|
2020-09-10 00:55:08 +08:00
|
|
|
int rc = (*WaitForMessage_f)(mRfb, 500);
|
|
|
|
if ( rc < 0 ) {
|
|
|
|
return -1;
|
|
|
|
} else if ( !rc ) {
|
|
|
|
return rc;
|
|
|
|
}
|
2020-04-14 05:16:01 +08:00
|
|
|
rfbBool res = (*HandleRFBServerMessage_f)(mRfb);
|
2020-11-18 00:08:11 +08:00
|
|
|
Debug(3, "PreCapture rc from HandleMessage %d", res == TRUE ? 1 : -1);
|
2020-09-09 22:25:55 +08:00
|
|
|
return res == TRUE ? 1 : -1;
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
2020-12-08 05:26:26 +08:00
|
|
|
int VncCamera::Capture(ZMPacket &zm_packet) {
|
2020-11-18 00:08:11 +08:00
|
|
|
if ( ! mVncData.buffer ) {
|
|
|
|
return 0;
|
|
|
|
}
|
2020-12-08 05:26:26 +08:00
|
|
|
uint8_t *directbuffer = zm_packet.image->WriteBuffer(width, height, colours, subpixelorder);
|
2020-11-18 00:08:11 +08:00
|
|
|
int rc = scale.Convert(
|
2020-09-09 22:25:55 +08:00
|
|
|
mVncData.buffer,
|
|
|
|
mRfb->si.framebufferHeight * mRfb->si.framebufferWidth * 4,
|
|
|
|
directbuffer,
|
|
|
|
width * height * mBpp,
|
|
|
|
AV_PIX_FMT_RGBA,
|
|
|
|
mImgPixFmt,
|
|
|
|
mRfb->si.framebufferWidth,
|
|
|
|
mRfb->si.framebufferHeight,
|
|
|
|
width,
|
|
|
|
height);
|
2020-11-18 00:08:11 +08:00
|
|
|
return rc == 0 ? 1 : rc;
|
2020-03-26 14:21:34 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
int VncCamera::PostCapture() {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
int VncCamera::Close() {
|
|
|
|
return 0;
|
2020-03-26 15:08:00 +08:00
|
|
|
}
|
2020-03-28 23:59:05 +08:00
|
|
|
#endif
|