obs-StreamFX/source/filters/filter-displacement.cpp
Michael Fabian 'Xaymar' Dirks bbcde070f3 filter-displacement: Do not allow zero-length file names
Fixes a crash when using c_str() on an empty string, causing an unexpected crash.
2019-04-23 17:00:18 +02:00

288 lines
9 KiB
C++

/*
* Modern effects for a modern Streamer
* Copyright (C) 2017 Michael Fabian Dirks
*
* This program 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 program 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 program; if not, write to the Free Software
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA
*/
#include "filter-displacement.hpp"
#include <sys/stat.h>
#include "strings.hpp"
// Initializer & Finalizer
INITIALIZER(FilterDisplacementInit)
{
initializerFunctions.push_back([] { filter::displacement::displacement_factory::initialize(); });
finalizerFunctions.push_back([] { filter::displacement::displacement_factory::finalize(); });
}
static std::shared_ptr<filter::displacement::displacement_factory> factory_instance = nullptr;
void filter::displacement::displacement_factory::initialize()
{
factory_instance = std::make_shared<filter::displacement::displacement_factory>();
}
void filter::displacement::displacement_factory::finalize()
{
factory_instance.reset();
}
std::shared_ptr<filter::displacement::displacement_factory> filter::displacement::displacement_factory::get()
{
return factory_instance;
}
filter::displacement::displacement_factory::displacement_factory()
{
memset(&sourceInfo, 0, sizeof(obs_source_info));
sourceInfo.id = "obs-stream-effects-filter-displacement";
sourceInfo.type = OBS_SOURCE_TYPE_FILTER;
sourceInfo.output_flags = OBS_SOURCE_VIDEO;
sourceInfo.get_name = get_name;
sourceInfo.get_defaults = get_defaults;
sourceInfo.get_properties = get_properties;
sourceInfo.create = create;
sourceInfo.destroy = destroy;
sourceInfo.update = update;
sourceInfo.activate = activate;
sourceInfo.deactivate = deactivate;
sourceInfo.show = show;
sourceInfo.hide = hide;
sourceInfo.video_tick = video_tick;
sourceInfo.video_render = video_render;
obs_register_source(&sourceInfo);
}
filter::displacement::displacement_factory::~displacement_factory() {}
const char* filter::displacement::displacement_factory::get_name(void*)
{
return P_TRANSLATE(S_FILTER_DISPLACEMENT);
}
void* filter::displacement::displacement_factory::create(obs_data_t* data, obs_source_t* source)
{
return new displacement_instance(data, source);
}
void filter::displacement::displacement_factory::destroy(void* ptr)
{
delete reinterpret_cast<displacement_instance*>(ptr);
}
uint32_t filter::displacement::displacement_factory::get_width(void* ptr)
{
return reinterpret_cast<displacement_instance*>(ptr)->get_width();
}
uint32_t filter::displacement::displacement_factory::get_height(void* ptr)
{
return reinterpret_cast<displacement_instance*>(ptr)->get_height();
}
void filter::displacement::displacement_factory::get_defaults(obs_data_t* data)
{
char* disp = obs_module_file("filter-displacement/neutral.png");
obs_data_set_default_string(data, S_FILTER_DISPLACEMENT_FILE, disp);
obs_data_set_default_double(data, S_FILTER_DISPLACEMENT_RATIO, 0);
obs_data_set_default_double(data, S_FILTER_DISPLACEMENT_SCALE, 0);
bfree(disp);
}
obs_properties_t* filter::displacement::displacement_factory::get_properties(void* ptr)
{
obs_properties_t* pr = obs_properties_create();
std::string path = "";
if (ptr)
path = reinterpret_cast<displacement_instance*>(ptr)->get_file();
obs_properties_add_path(pr, S_FILTER_DISPLACEMENT_FILE, P_TRANSLATE(S_FILTER_DISPLACEMENT_FILE),
obs_path_type::OBS_PATH_FILE, P_TRANSLATE(S_FILTER_DISPLACEMENT_FILE_TYPES), path.c_str());
obs_properties_add_float_slider(pr, S_FILTER_DISPLACEMENT_RATIO, P_TRANSLATE(S_FILTER_DISPLACEMENT_RATIO), 0, 1,
0.01);
obs_properties_add_float_slider(pr, S_FILTER_DISPLACEMENT_SCALE, P_TRANSLATE(S_FILTER_DISPLACEMENT_SCALE), -1000,
1000, 0.01);
return pr;
}
void filter::displacement::displacement_factory::update(void* ptr, obs_data_t* data)
{
reinterpret_cast<displacement_instance*>(ptr)->update(data);
}
void filter::displacement::displacement_factory::activate(void* ptr)
{
reinterpret_cast<displacement_instance*>(ptr)->activate();
}
void filter::displacement::displacement_factory::deactivate(void* ptr)
{
reinterpret_cast<displacement_instance*>(ptr)->deactivate();
}
void filter::displacement::displacement_factory::show(void* ptr)
{
reinterpret_cast<displacement_instance*>(ptr)->show();
}
void filter::displacement::displacement_factory::hide(void* ptr)
{
reinterpret_cast<displacement_instance*>(ptr)->hide();
}
void filter::displacement::displacement_factory::video_tick(void* ptr, float time)
{
reinterpret_cast<displacement_instance*>(ptr)->video_tick(time);
}
void filter::displacement::displacement_factory::video_render(void* ptr, gs_effect_t* effect)
{
reinterpret_cast<displacement_instance*>(ptr)->video_render(effect);
}
void filter::displacement::displacement_instance::validate_file_texture(std::string file)
{
bool do_update = false;
// Don't allow empty file names.
if (file.length() == 0) {
return;
}
// File name different
if (file != m_file_name) {
do_update = true;
m_file_name = file;
}
// Timestamp verification
struct stat stats;
if (os_stat(m_file_name.c_str(), &stats) != 0) {
do_update = do_update || (stats.st_ctime != m_file_create_time);
do_update = do_update || (stats.st_mtime != m_file_modified_time);
do_update = do_update || (static_cast<size_t>(stats.st_size) != m_file_size);
m_file_create_time = stats.st_ctime;
m_file_modified_time = stats.st_mtime;
m_file_size = static_cast<size_t>(stats.st_size);
}
do_update = !m_file_texture || do_update;
if (do_update) {
m_file_texture = std::make_shared<gs::texture>(m_file_name);
}
}
filter::displacement::displacement_instance::displacement_instance(obs_data_t* data, obs_source_t* context)
: m_self(context), m_timer(0), m_effect(nullptr), m_distance(0), m_file_create_time(0), m_file_modified_time(0),
m_file_size(0)
{
char* effectFile = obs_module_file("effects/displace.effect");
try {
m_effect = std::make_shared<gs::effect>(effectFile);
} catch (...) {
P_LOG_ERROR("<Displacement Filter:%s> Failed to load displacement effect.", obs_source_get_name(m_self));
}
bfree(effectFile);
update(data);
}
filter::displacement::displacement_instance::~displacement_instance()
{
m_effect.reset();
m_file_texture.reset();
}
void filter::displacement::displacement_instance::update(obs_data_t* data)
{
validate_file_texture(obs_data_get_string(data, S_FILTER_DISPLACEMENT_FILE));
m_distance = float_t(obs_data_get_double(data, S_FILTER_DISPLACEMENT_RATIO));
vec2_set(&m_displacement_scale, float_t(obs_data_get_double(data, S_FILTER_DISPLACEMENT_SCALE)),
float_t(obs_data_get_double(data, S_FILTER_DISPLACEMENT_SCALE)));
}
uint32_t filter::displacement::displacement_instance::get_width()
{
return 0;
}
uint32_t filter::displacement::displacement_instance::get_height()
{
return 0;
}
void filter::displacement::displacement_instance::activate() {}
void filter::displacement::displacement_instance::deactivate() {}
void filter::displacement::displacement_instance::show() {}
void filter::displacement::displacement_instance::hide() {}
void filter::displacement::displacement_instance::video_tick(float time)
{
m_timer += time;
if (m_timer >= 1.0f) {
m_timer -= 1.0f;
validate_file_texture(m_file_name);
}
}
static float interp(float a, float b, float v)
{
return (a * (1.0f - v)) + (b * v);
}
void filter::displacement::displacement_instance::video_render(gs_effect_t*)
{
obs_source_t* parent = obs_filter_get_parent(m_self);
obs_source_t* target = obs_filter_get_target(m_self);
uint32_t baseW = obs_source_get_base_width(target), baseH = obs_source_get_base_height(target);
// Skip rendering if our target, parent or context is not valid.
if (!parent || !target || !baseW || !baseH || !m_file_texture) {
obs_source_skip_video_filter(m_self);
return;
}
if (!obs_source_process_filter_begin(m_self, GS_RGBA, OBS_ALLOW_DIRECT_RENDERING)) {
obs_source_skip_video_filter(m_self);
return;
}
if (m_effect->has_parameter("texelScale")) {
m_effect->get_parameter("texelScale")
.set_float2(interp((1.0f / baseW), 1.0f, m_distance), interp((1.0f / baseH), 1.0f, m_distance));
}
if (m_effect->has_parameter("displacementScale")) {
m_effect->get_parameter("displacementScale").set_float2(m_displacement_scale);
}
if (m_effect->has_parameter("displacementMap")) {
m_effect->get_parameter("displacementMap").set_texture(m_file_texture);
}
obs_source_process_filter_end(m_self, m_effect->get_object(), baseW, baseH);
}
std::string filter::displacement::displacement_instance::get_file()
{
return m_file_name;
}