mirror of
https://github.com/outbackdingo/UltraGrid.git
synced 2026-03-21 21:40:20 +00:00
624 lines
21 KiB
C++
624 lines
21 KiB
C++
/**
|
|
* @file video_compress.c
|
|
* @author Martin Pulec <pulec@cesnet.cz>
|
|
* @ingroup video_compress
|
|
*
|
|
* @brief Video compress functions.
|
|
*/
|
|
/*
|
|
* Copyright (c) 2011-2013 CESNET z.s.p.o.
|
|
* All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without
|
|
* modification, is permitted provided that the following conditions
|
|
* are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright
|
|
* notice, this list of conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
* documentation and/or other materials provided with the distribution.
|
|
*
|
|
* 3. Neither the name of CESNET nor the names of its contributors may be
|
|
* used to endorse or promote products derived from this software without
|
|
* specific prior written permission.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY THE AUTHORS AND CONTRIBUTORS
|
|
* "AS IS" AND ANY EXPRESSED OR IMPLIED WARRANTIES, INCLUDING,
|
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY
|
|
* AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
|
* EVENT SHALL THE AUTHORS OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT,
|
|
* INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
|
|
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
|
* OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE,
|
|
* EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
#include "config.h"
|
|
#include "config_unix.h"
|
|
#include "config_win32.h"
|
|
#endif // HAVE_CONFIG_H
|
|
|
|
#include <list>
|
|
#include <memory>
|
|
#include <stdio.h>
|
|
#include <string>
|
|
#include <string.h>
|
|
#include <tuple>
|
|
#include <vector>
|
|
|
|
#include "messaging.h"
|
|
#include "module.h"
|
|
#include "utils/synchronized_queue.h"
|
|
#include "utils/vf_split.h"
|
|
#include "utils/worker.h"
|
|
#include "video.h"
|
|
#include "video_compress.h"
|
|
#include "video_compress/cuda_dxt.h"
|
|
#include "video_compress/dxt_glsl.h"
|
|
#include "video_compress/libavcodec.h"
|
|
#include "video_compress/jpeg.h"
|
|
#include "video_compress/none.h"
|
|
#include "video_compress/uyvy.h"
|
|
#include "lib_common.h"
|
|
|
|
using namespace std;
|
|
|
|
namespace {
|
|
|
|
/* *_str are symbol names inside library */
|
|
/**
|
|
* @brief This struct describes individual compress module
|
|
*
|
|
* Initially, in this struct are either callbacks or functions names.
|
|
* For actual initialization of the callbacks/names, @ref MK_STATIC and @ref MK_NAME
|
|
* macros should be used. After initialization, callbacks are set.
|
|
*
|
|
* There are 2 APIs available - drivers are required to implement one of them. They
|
|
* can implement either @ref compress_frame_func or @ref compress_tile_func function.
|
|
* The other* shall then be NULL.
|
|
*/
|
|
struct compress_t {
|
|
const char * library_name; ///< If module is dynamically loadable, this is the name of library.
|
|
|
|
compress_info_t * compress_info;
|
|
const char * compress_info_symbol_name;
|
|
|
|
void *handle; ///< for modular build, dynamically loaded library handle
|
|
};
|
|
|
|
/**
|
|
* @brief This structure represents real internal compress state
|
|
*/
|
|
struct compress_state_real {
|
|
struct compress_t *handle; ///< handle for the driver
|
|
struct module **state; ///< driver internal states
|
|
unsigned int state_count; ///< count of compress states (equal to tiles' count)
|
|
char compress_options[1024]; ///< compress options (for reconfiguration)
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @brief Video compress state.
|
|
*
|
|
* This structure represents external video compress state. This is basically a proxy for real
|
|
* state. The point of doing this is to allow dynamic reconfiguration of the real state.
|
|
*/
|
|
struct compress_state {
|
|
struct module mod; ///< compress module data
|
|
struct compress_state_real *ptr; ///< pointer to real compress state
|
|
synchronized_queue<shared_ptr<video_frame>, 1> queue;
|
|
};
|
|
|
|
typedef struct compress_state compress_state_proxy; ///< Used to emphasize that the state is actually a proxy.
|
|
|
|
/**
|
|
* This is placeholder state returned by compression module meaning that the initialization was
|
|
* successful but no state was create. This is the case eg. when the module only displayed help.
|
|
*/
|
|
struct module compress_init_noerr;
|
|
|
|
static void init_compressions(void);
|
|
static shared_ptr<video_frame> compress_frame_tiles(struct compress_state_real *s,
|
|
shared_ptr<video_frame> frame, struct module *parent);
|
|
static int compress_init_real(struct module *parent, const char *config_string,
|
|
struct compress_state_real **state);
|
|
static void compress_done_real(struct compress_state_real *s);
|
|
static void compress_done(struct module *mod);
|
|
|
|
/**
|
|
* @brief This table contains list of video compress devices compiled with this UltraGrid version.
|
|
* @copydetails decoders
|
|
*/
|
|
struct compress_t compress_modules[] = {
|
|
#ifndef UV_IN_YURI
|
|
#if defined HAVE_DXT_GLSL || defined BUILD_LIBRARIES
|
|
{
|
|
"rtdxt",
|
|
MK_NAME_REF(rtdxt_info),
|
|
NULL,
|
|
},
|
|
#endif
|
|
#if defined HAVE_JPEG || defined BUILD_LIBRARIES
|
|
{
|
|
"jpeg",
|
|
MK_NAME_REF(jpeg_info),
|
|
NULL,
|
|
},
|
|
#endif
|
|
#if defined HAVE_COMPRESS_UYVY || defined BUILD_LIBRARIES
|
|
{
|
|
"uyvy",
|
|
MK_NAME_REF(uyvy_info),
|
|
NULL,
|
|
},
|
|
#endif
|
|
#if defined HAVE_LAVC || defined BUILD_LIBRARIES
|
|
{
|
|
"libavcodec",
|
|
MK_NAME_REF(libavcodec_info),
|
|
NULL,
|
|
},
|
|
#endif
|
|
#if defined HAVE_CUDA_DXT || defined BUILD_LIBRARIES
|
|
{
|
|
"cuda_dxt",
|
|
MK_NAME_REF(cuda_dxt_info),
|
|
NULL,
|
|
},
|
|
#endif
|
|
#endif
|
|
{
|
|
NULL,
|
|
MK_STATIC_REF(none_info),
|
|
NULL,
|
|
},
|
|
};
|
|
|
|
#define MAX_COMPRESS_MODULES (sizeof(compress_modules)/sizeof(struct compress_t))
|
|
|
|
/// @brief List of available display devices.
|
|
///
|
|
/// initialized automatically
|
|
static struct compress_t *available_compress_modules[MAX_COMPRESS_MODULES];
|
|
/// @brief Count of @ref available_compress_modules.
|
|
///
|
|
/// initialized automatically
|
|
static int compress_modules_count = 0;
|
|
|
|
#ifdef BUILD_LIBRARIES
|
|
#include <dlfcn.h>
|
|
/** Opens compress library of given name. */
|
|
static void *compress_open_library(const char *compress_name)
|
|
{
|
|
char name[128];
|
|
snprintf(name, sizeof(name), "vcompress_%s.so.%d", compress_name, VIDEO_COMPRESS_ABI_VERSION);
|
|
|
|
return open_library(name);
|
|
}
|
|
|
|
/** For a given device, load individual functions from library handle (previously opened). */
|
|
static int compress_fill_symbols(struct compress_t *compression)
|
|
{
|
|
void *handle = compression->handle;
|
|
|
|
compression->compress_info = (compress_info_t *)
|
|
dlsym(handle, compression->compress_info_symbol_name);
|
|
|
|
if(!compression->compress_info) {
|
|
fprintf(stderr, "Library %s opening error: %s \n", compression->library_name, dlerror());
|
|
return FALSE;
|
|
}
|
|
return TRUE;
|
|
}
|
|
#endif
|
|
|
|
/// @brief guard of @ref available_compress_modules initialization
|
|
static pthread_once_t compression_list_initialized = PTHREAD_ONCE_INIT;
|
|
|
|
/// @brief initializes @ref available_compress_modules initialization
|
|
static void init_compressions(void)
|
|
{
|
|
unsigned int i;
|
|
for(i = 0; i < sizeof(compress_modules)/sizeof(struct compress_t); ++i) {
|
|
#ifdef BUILD_LIBRARIES
|
|
if(compress_modules[i].library_name) {
|
|
int ret;
|
|
compress_modules[i].handle = compress_open_library(
|
|
compress_modules[i].library_name);
|
|
if(!compress_modules[i].handle) continue;
|
|
ret = compress_fill_symbols(&compress_modules[i]);
|
|
if(!ret) {
|
|
fprintf(stderr, "Opening symbols from library %s failed.\n",
|
|
compress_modules[i].library_name);
|
|
continue;
|
|
}
|
|
}
|
|
#endif
|
|
available_compress_modules[compress_modules_count] = &compress_modules[i];
|
|
compress_modules_count++;
|
|
}
|
|
}
|
|
|
|
/// @brief Displays list of available compressions.
|
|
void show_compress_help()
|
|
{
|
|
int i;
|
|
pthread_once(&compression_list_initialized, init_compressions);
|
|
printf("Possible compression modules (see '-c <module>:help' for options):\n");
|
|
for(i = 0; i < compress_modules_count; ++i) {
|
|
printf("\t%s\n", available_compress_modules[i]->compress_info->name);
|
|
}
|
|
}
|
|
|
|
list<compress_preset> get_compress_capabilities()
|
|
{
|
|
list<compress_preset> ret;
|
|
|
|
pthread_once(&compression_list_initialized, init_compressions);
|
|
|
|
for (int i = 0; i < compress_modules_count; ++i) {
|
|
for (auto const & it : available_compress_modules[i]->compress_info->presets) {
|
|
auto new_elem = it;
|
|
new_elem.name = string(available_compress_modules[i]->compress_info->name)
|
|
+ ":" + it.name;
|
|
if (available_compress_modules[i]->compress_info->is_supported_func &&
|
|
available_compress_modules[i]->compress_info->is_supported_func()) {
|
|
ret.push_back(new_elem);
|
|
}
|
|
}
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief Processes message.
|
|
*
|
|
* This function is a callback called from control thread to change some parameters of
|
|
* compression.
|
|
*
|
|
* @param[in] receiver pointer to the compress module
|
|
* @param[in] msg message to process
|
|
*/
|
|
static void compress_process_message(compress_state_proxy *proxy, struct msg_change_compress_data *data)
|
|
{
|
|
/* In this case we are only changing some parameter of compression.
|
|
* This means that we pass the parameter to compress driver. */
|
|
if(data->what == CHANGE_PARAMS) {
|
|
for(unsigned int i = 0; i < proxy->ptr->state_count; ++i) {
|
|
struct msg_change_compress_data *tmp_data =
|
|
(struct msg_change_compress_data *)
|
|
new_message(sizeof(struct msg_change_compress_data));
|
|
tmp_data->what = data->what;
|
|
strncpy(tmp_data->config_string, data->config_string,
|
|
sizeof(tmp_data->config_string) - 1);
|
|
struct response *resp = send_message_to_receiver(proxy->ptr->state[i],
|
|
(struct message *) tmp_data);
|
|
resp->deleter(resp);
|
|
}
|
|
|
|
} else {
|
|
struct compress_state_real *new_state;
|
|
char config[1024];
|
|
strncpy(config, data->config_string, sizeof(config));
|
|
|
|
int ret = compress_init_real(&proxy->mod, config, &new_state);
|
|
if(ret == 0) {
|
|
struct compress_state_real *old = proxy->ptr;
|
|
proxy->ptr = new_state;
|
|
compress_done_real(old);
|
|
}
|
|
}
|
|
|
|
free_message((struct message *) data);
|
|
}
|
|
|
|
/**
|
|
* @brief This function initializes video compression.
|
|
*
|
|
* This function wrapps the call of compress_init_real().
|
|
* @param[in] parent parent module
|
|
* @param[in] config_string configuration (in format <driver>:<options>)
|
|
* @param[out] state created state
|
|
* @retval 0 if state created sucessfully
|
|
* @retval <0 if error occured
|
|
* @retval >0 finished successfully, no state created (eg. displayed help)
|
|
*/
|
|
int compress_init(struct module *parent, const char *config_string, struct compress_state **state) {
|
|
struct compress_state_real *s;
|
|
|
|
compress_state_proxy *proxy;
|
|
proxy = new compress_state_proxy();
|
|
|
|
module_init_default(&proxy->mod);
|
|
proxy->mod.cls = MODULE_CLASS_COMPRESS;
|
|
proxy->mod.priv_data = proxy;
|
|
proxy->mod.deleter = compress_done;
|
|
|
|
int ret = compress_init_real(&proxy->mod, config_string, &s);
|
|
if(ret == 0) {
|
|
proxy->ptr = s;
|
|
|
|
*state = proxy;
|
|
module_register(&proxy->mod, parent);
|
|
} else {
|
|
delete proxy;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
/**
|
|
* @brief This is compression initialization function that really does the stuff.
|
|
* @param[in] parent parent module
|
|
* @param[in] config_string configuration (in format <driver>:<options>)
|
|
* @param[out] state created state
|
|
* @retval 0 if state created sucessfully
|
|
* @retval <0 if error occured
|
|
* @retval >0 finished successfully, no state created (eg. displayed help)
|
|
*/
|
|
static int compress_init_real(struct module *parent, const char *config_string,
|
|
struct compress_state_real **state)
|
|
{
|
|
struct compress_state_real *s;
|
|
struct video_compress_params params;
|
|
const char *compress_options = NULL;
|
|
|
|
if(!config_string)
|
|
return -1;
|
|
|
|
if(strcmp(config_string, "help") == 0)
|
|
{
|
|
show_compress_help();
|
|
return 1;
|
|
}
|
|
|
|
pthread_once(&compression_list_initialized, init_compressions);
|
|
|
|
memset(¶ms, 0, sizeof(params));
|
|
|
|
s = (struct compress_state_real *) calloc(1, sizeof(struct compress_state_real));
|
|
|
|
s->state_count = 1;
|
|
|
|
for (int i = 0; i < compress_modules_count; ++i) {
|
|
if(strncasecmp(config_string, available_compress_modules[i]->compress_info->name,
|
|
strlen(available_compress_modules[i]->compress_info->name)) == 0) {
|
|
s->handle = available_compress_modules[i];
|
|
if (config_string[strlen(available_compress_modules[i]->compress_info->name)] == ':')
|
|
compress_options = config_string +
|
|
strlen(available_compress_modules[i]->compress_info->name) + 1;
|
|
else
|
|
compress_options = "";
|
|
}
|
|
}
|
|
if(!s->handle) {
|
|
fprintf(stderr, "Unknown compression: %s\n", config_string);
|
|
free(s);
|
|
return -1;
|
|
}
|
|
strncpy(s->compress_options, compress_options, sizeof(s->compress_options) - 1);
|
|
s->compress_options[sizeof(s->compress_options) - 1] = '\0';
|
|
params.cfg = s->compress_options;
|
|
if(s->handle->compress_info->init_func) {
|
|
s->state = (struct module **) calloc(1, sizeof(struct module *));
|
|
s->state[0] = s->handle->compress_info->init_func(parent, ¶ms);
|
|
if(!s->state[0]) {
|
|
fprintf(stderr, "Compression initialization failed: %s\n", config_string);
|
|
free(s->state);
|
|
free(s);
|
|
return -1;
|
|
}
|
|
if(s->state[0] == &compress_init_noerr) {
|
|
free(s->state);
|
|
free(s);
|
|
return 1;
|
|
}
|
|
} else {
|
|
free(s);
|
|
return -1;
|
|
}
|
|
|
|
*state = s;
|
|
return 0;
|
|
}
|
|
|
|
/**
|
|
* @brief Returns name of compression module
|
|
*
|
|
* @param proxy compress state
|
|
* @returns compress name
|
|
*/
|
|
const char *get_compress_name(compress_state_proxy *proxy)
|
|
{
|
|
if(proxy)
|
|
return proxy->ptr->handle->compress_info->name;
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
/**
|
|
* @brief Compressses frame
|
|
*
|
|
* @param proxy compress state
|
|
* @param frame uncompressed frame to be compressed
|
|
* @return compressed frame, may be NULL if compression failed
|
|
*/
|
|
void compress_frame(compress_state_proxy *proxy, shared_ptr<video_frame> frame)
|
|
{
|
|
if (!frame) { // pass poisoned pill
|
|
if (proxy)
|
|
proxy->queue.push(shared_ptr<video_frame>());
|
|
return;
|
|
}
|
|
|
|
if (!proxy)
|
|
abort();
|
|
|
|
struct msg_change_compress_data *msg = NULL;
|
|
while ((msg = (struct msg_change_compress_data *) check_message(&proxy->mod))) {
|
|
compress_process_message(proxy, msg);
|
|
}
|
|
|
|
struct compress_state_real *s = proxy->ptr;
|
|
|
|
shared_ptr<video_frame> sync_api_frame;
|
|
if (s->handle->compress_info->compress_frame_func) {
|
|
sync_api_frame = s->handle->compress_info->compress_frame_func(s->state[0], frame);
|
|
} else if(s->handle->compress_info->compress_tile_func) {
|
|
sync_api_frame = compress_frame_tiles(s, frame, &proxy->mod);
|
|
} else {
|
|
sync_api_frame = {};
|
|
}
|
|
|
|
// empty return value here represents error, but we don't want to pass it to queue, since it would
|
|
// be interpreted as poisoned pill
|
|
if (!sync_api_frame) {
|
|
return;
|
|
}
|
|
|
|
proxy->queue.push(sync_api_frame);
|
|
}
|
|
|
|
/**
|
|
* @name Tile API Routines
|
|
* The worker callbacks here are optimization - all tiles are processed concurrently.
|
|
* @{
|
|
*/
|
|
/**
|
|
* @brief Auxiliary structure passed to worker thread.
|
|
*/
|
|
struct compress_worker_data {
|
|
struct module *state; ///< compress driver status
|
|
shared_ptr<video_frame> frame; ///< uncompressed tile to be compressed
|
|
|
|
compress_tile_t callback; ///< tile compress callback
|
|
shared_ptr<video_frame> ret; ///< OUT - returned compressed tile, NULL if failed
|
|
};
|
|
|
|
/**
|
|
* @brief This function is callback passed to a "thread pool"
|
|
* @param arg @ref compress_worker_data
|
|
* @return @ref compress_worker_data (same as input)
|
|
*/
|
|
static void *compress_tile_callback(void *arg) {
|
|
compress_worker_data *s = (compress_worker_data *) arg;
|
|
|
|
s->ret = s->callback(s->state, s->frame);
|
|
|
|
return s;
|
|
}
|
|
|
|
/**
|
|
* Compresses video frame with tiles API
|
|
*
|
|
* @param[in] s compress state
|
|
* @param[in] frame uncompressed frame
|
|
* @param parent parent module (for the case when there is a need to reconfigure)
|
|
* @return compressed video frame, may be NULL if compression failed
|
|
*/
|
|
static shared_ptr<video_frame> compress_frame_tiles(struct compress_state_real *s,
|
|
shared_ptr<video_frame> frame, struct module *parent)
|
|
{
|
|
struct video_compress_params params;
|
|
memset(¶ms, 0, sizeof(params));
|
|
params.cfg = s->compress_options;
|
|
if(frame->tile_count != s->state_count) {
|
|
s->state = (struct module **) realloc(s->state, frame->tile_count * sizeof(struct module *));
|
|
for(unsigned int i = s->state_count; i < frame->tile_count; ++i) {
|
|
s->state[i] = s->handle->compress_info->init_func(parent, ¶ms);
|
|
if(!s->state[i]) {
|
|
fprintf(stderr, "Compression initialization failed\n");
|
|
return NULL;
|
|
}
|
|
}
|
|
s->state_count = frame->tile_count;
|
|
}
|
|
|
|
vector<shared_ptr<video_frame>> separate_tiles = vf_separate_tiles(frame);
|
|
// frame pointer may no longer be valid
|
|
frame = NULL;
|
|
|
|
vector<task_result_handle_t> task_handle(separate_tiles.size());
|
|
|
|
vector <compress_worker_data> data_tile(separate_tiles.size());
|
|
for(unsigned int i = 0; i < separate_tiles.size(); ++i) {
|
|
struct compress_worker_data *data = &data_tile[i];
|
|
data->state = s->state[i];
|
|
data->frame = separate_tiles[i];
|
|
data->callback = s->handle->compress_info->compress_tile_func;
|
|
|
|
task_handle[i] = task_run_async(compress_tile_callback, data);
|
|
}
|
|
|
|
vector<shared_ptr<video_frame>> compressed_tiles(separate_tiles.size(), nullptr);
|
|
|
|
bool failed = false;
|
|
for(unsigned int i = 0; i < separate_tiles.size(); ++i) {
|
|
struct compress_worker_data *data = (struct compress_worker_data *)
|
|
wait_task(task_handle[i]);
|
|
|
|
if(!data->ret) {
|
|
failed = true;
|
|
}
|
|
|
|
compressed_tiles[i] = data->ret;
|
|
}
|
|
|
|
if (failed) {
|
|
return NULL;
|
|
}
|
|
|
|
return vf_merge_tiles(compressed_tiles);
|
|
}
|
|
/**
|
|
* @}
|
|
*/
|
|
|
|
/**
|
|
* @brief Video compression cleanup function.
|
|
* @param mod video compress module
|
|
*/
|
|
static void compress_done(struct module *mod)
|
|
{
|
|
if(!mod)
|
|
return;
|
|
|
|
compress_state_proxy *proxy = (compress_state_proxy *) mod->priv_data;
|
|
struct compress_state_real *s = proxy->ptr;
|
|
compress_done_real(s);
|
|
|
|
delete proxy;
|
|
}
|
|
|
|
/**
|
|
* Video compression cleanup function.
|
|
* This destroys contained real video compress state.
|
|
* @param mod video compress module
|
|
*/
|
|
static void compress_done_real(struct compress_state_real *s)
|
|
{
|
|
if(!s)
|
|
return;
|
|
|
|
for(unsigned int i = 0; i < s->state_count; ++i) {
|
|
module_done(s->state[i]);
|
|
}
|
|
free(s->state);
|
|
free(s);
|
|
}
|
|
|
|
shared_ptr<video_frame> compress_pop(compress_state_proxy *proxy)
|
|
{
|
|
if(!proxy)
|
|
return NULL;
|
|
|
|
return proxy->queue.pop();
|
|
}
|
|
|