zoneminder/src/zm_libvnc_camera.cpp

198 lines
5.3 KiB
C++
Raw Normal View History

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
#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() {
if ( libvnc_lib != nullptr ) // Safe-check
2020-04-14 05:16:01 +08:00
return;
libvnc_lib = dlopen("libvncclient.so", RTLD_LAZY | RTLD_GLOBAL);
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;
}
static char* GetPasswordCallback(rfbClient* cl){
2020-04-14 05:16:01 +08:00
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));
if ( credentialType != rfbCredentialTypeUser ) {
free(c);
return nullptr;
2020-03-26 14:21:34 +08:00
}
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
),
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);
}
if ( capture )
2020-03-26 14:21:34 +08:00
Initialise();
}
VncCamera::~VncCamera() {
if ( capture )
2020-03-26 14:21:34 +08:00
Terminate();
}
void VncCamera::Initialise() {
Debug(2, "Initializing Client");
2020-04-14 05:16:01 +08:00
bind_libvnc_symbols();
mRfb = (*rfbGetClient_f)(8, 3, 4);
2020-03-28 02:15:15 +08:00
2020-04-14 05:16:01 +08:00
(*rfbClientSetClientData_f)(mRfb, &TAG_0, &mVncData);
(*rfbClientSetClientData_f)(mRfb, &TAG_1, (void *)mPass.c_str());
(*rfbClientSetClientData_f)(mRfb, &TAG_2, (void *)mUser.c_str());
2020-03-26 14:21:34 +08:00
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-03-29 06:17:19 +08:00
scale.init();
2020-03-26 14:21:34 +08:00
}
void VncCamera::Terminate() {
if ( mRfb->frameBuffer )
free(mRfb->frameBuffer);
2020-04-14 05:16:01 +08:00
(*rfbClientCleanup_f)(mRfb);
2020-03-26 14:21:34 +08:00
return;
}
int VncCamera::PrimeCapture() {
Debug(1, "Priming capture from %s", mHost.c_str());
2020-09-10 00:55:08 +08:00
if ( ! (*rfbInitClient_f)(mRfb, 0, nullptr) ) {
return -1;
}
2020-03-26 14:21:34 +08:00
return 0;
}
int VncCamera::PreCapture() {
2020-03-28 23:59:05 +08:00
Debug(2, "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);
return res == TRUE ? 1 : -1;
2020-03-26 14:21:34 +08:00
}
int VncCamera::Capture(Image &image) {
Debug(2, "Capturing");
2020-03-29 06:17:19 +08:00
uint8_t *directbuffer = image.WriteBuffer(width, height, colours, subpixelorder);
scale.Convert(
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-03-26 14:21:34 +08:00
return 1;
}
int VncCamera::PostCapture() {
return 0;
}
int VncCamera::CaptureAndRecord(Image &image, timeval recording, char* event_directory) {
return 0;
}
int VncCamera::Close() {
return 0;
}
2020-03-28 23:59:05 +08:00
#endif