Files
obs-studio/libobs/obs-output.c
jp9000 3d6d43225f Add planar audio support, improve test output
- Add planar audio support.  FFmpeg and libav use planar audio for many
  encoders, so it was somewhat necessary to add support in libobs
  itself.

- Improve/adjust FFmpeg test output plugin.  The exports were somewhat
  messed up (making me rethink how exports should be done).  Not yet
  functional; it handles video properly, but it still does not handle
  audio properly.

- Improve planar video code.  The planar video code was not properly
  accounting for row sizes for each plane.  Specifying row sizes for
  each plane has now been added.  This will also make it more compatible
  with FFmpeg/libav.

- Fixed a bug where callbacks wouldn't create properly in audio-io and
  video-io code.

- Implement 'blogva' function to allow for va_list usage with libobs
  logging.
2014-02-07 03:03:54 -07:00

139 lines
3.6 KiB
C

/******************************************************************************
Copyright (C) 2013 by Hugh Bailey <obs.jim@gmail.com>
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, see <http://www.gnu.org/licenses/>.
******************************************************************************/
#include "obs.h"
#include "obs-internal.h"
bool load_output_info(void *module, const char *module_name,
const char *id, struct output_info *info)
{
LOAD_MODULE_SUBFUNC(getname, true);
LOAD_MODULE_SUBFUNC(create, true);
LOAD_MODULE_SUBFUNC(destroy, true);
LOAD_MODULE_SUBFUNC(update, true);
LOAD_MODULE_SUBFUNC(start, true);
LOAD_MODULE_SUBFUNC(stop, true);
LOAD_MODULE_SUBFUNC(active, true);
LOAD_MODULE_SUBFUNC(properties, false);
LOAD_MODULE_SUBFUNC(pause, false);
info->id = id;
return true;
}
static inline const struct output_info *find_output(const char *id)
{
size_t i;
for (i = 0; i < obs->output_types.num; i++)
if (strcmp(obs->output_types.array[i].id, id) == 0)
return obs->output_types.array+i;
return NULL;
}
obs_output_t obs_output_create(const char *id, const char *name,
obs_data_t settings)
{
const struct output_info *info = find_output(id);
struct obs_output *output;
if (!info) {
blog(LOG_WARNING, "Output '%s' not found", id);
return NULL;
}
output = bmalloc(sizeof(struct obs_output));
output->callbacks = *info;
output->settings = obs_data_newref(settings);
output->data = info->create(output->settings, output);
if (!output->data) {
obs_data_release(output->settings);
bfree(output);
return NULL;
}
output->name = bstrdup(name);
pthread_mutex_lock(&obs->data.outputs_mutex);
da_push_back(obs->data.outputs, &output);
pthread_mutex_unlock(&obs->data.outputs_mutex);
return output;
}
void obs_output_destroy(obs_output_t output)
{
if (output) {
if (output->callbacks.active) {
if (output->callbacks.active(output->data))
output->callbacks.stop(output->data);
}
pthread_mutex_lock(&obs->data.outputs_mutex);
da_erase_item(obs->data.outputs, &output);
pthread_mutex_unlock(&obs->data.outputs_mutex);
output->callbacks.destroy(output->data);
obs_data_release(output->settings);
bfree(output->name);
bfree(output);
}
}
bool obs_output_start(obs_output_t output)
{
return output->callbacks.start(output->data);
}
void obs_output_stop(obs_output_t output)
{
output->callbacks.stop(output->data);
}
bool obs_output_active(obs_output_t output)
{
return output->callbacks.active(output);
}
obs_properties_t obs_output_properties(const char *id, const char *locale)
{
const struct output_info *info = find_output(id);
if (info && info->properties)
return info->properties(locale);
return NULL;
}
void obs_output_update(obs_output_t output, obs_data_t settings)
{
obs_data_replace(&output->settings, settings);
if (output->callbacks.update)
output->callbacks.update(output->data, output->settings);
}
bool obs_output_canpause(obs_output_t output)
{
return output->callbacks.pause != NULL;
}
void obs_output_pause(obs_output_t output)
{
if (output->callbacks.pause)
output->callbacks.pause(output->data);
}