Revamp API and start using doxygen
The API used to be designed in such a way to where it would expect
exports for each individual source/output/encoder/etc. You would export
functions for each and it would automatically load those functions based
on a specific naming scheme from the module.
The idea behind this was that I wanted to limit the usage of structures
in the API so only functions could be used. It was an interesting idea
in theory, but this idea turned out to be flawed in a number of ways:
1.) Requiring exports to create sources/outputs/encoders/etc meant that
you could not create them by any other means, which meant that
things like faruton's .net plugin would become difficult.
2.) Export function declarations could not be checked, therefore if you
created a function with the wrong parameters and parameter types,
the compiler wouldn't know how to check for that.
3.) Required overly complex load functions in libobs just to handle it.
It makes much more sense to just have a load function that you call
manually. Complexity is the bane of all good programs.
4.) It required that you have functions of specific names, which looked
and felt somewhat unsightly.
So, to fix these issues, I replaced it with a more commonly used API
scheme, seen commonly in places like kernels and typical C libraries
with abstraction. You simply create a structure that contains the
callback definitions, and you pass it to a function to register that
definition (such as obs_register_source), which you call in the
obs_module_load of the module.
It will also automatically check the structure size and ensure that it
only loads the required values if the structure happened to add new
values in an API change.
The "main" source file for each module must include obs-module.h, and
must use OBS_DECLARE_MODULE() within that source file.
Also, started writing some doxygen documentation in to the main library
headers. Will add more detailed documentation as I go.
2014-02-12 08:04:50 -07:00
|
|
|
#include <obs-module.h>
|
2015-08-13 12:55:21 -07:00
|
|
|
#include <util/darray.h>
|
2015-08-04 08:52:12 -05:00
|
|
|
#include <libavutil/log.h>
|
2015-08-13 12:55:21 -07:00
|
|
|
#include <pthread.h>
|
2014-01-19 03:16:41 -07:00
|
|
|
|
Revamp API and start using doxygen
The API used to be designed in such a way to where it would expect
exports for each individual source/output/encoder/etc. You would export
functions for each and it would automatically load those functions based
on a specific naming scheme from the module.
The idea behind this was that I wanted to limit the usage of structures
in the API so only functions could be used. It was an interesting idea
in theory, but this idea turned out to be flawed in a number of ways:
1.) Requiring exports to create sources/outputs/encoders/etc meant that
you could not create them by any other means, which meant that
things like faruton's .net plugin would become difficult.
2.) Export function declarations could not be checked, therefore if you
created a function with the wrong parameters and parameter types,
the compiler wouldn't know how to check for that.
3.) Required overly complex load functions in libobs just to handle it.
It makes much more sense to just have a load function that you call
manually. Complexity is the bane of all good programs.
4.) It required that you have functions of specific names, which looked
and felt somewhat unsightly.
So, to fix these issues, I replaced it with a more commonly used API
scheme, seen commonly in places like kernels and typical C libraries
with abstraction. You simply create a structure that contains the
callback definitions, and you pass it to a function to register that
definition (such as obs_register_source), which you call in the
obs_module_load of the module.
It will also automatically check the structure size and ensure that it
only loads the required values if the structure happened to add new
values in an API change.
The "main" source file for each module must include obs-module.h, and
must use OBS_DECLARE_MODULE() within that source file.
Also, started writing some doxygen documentation in to the main library
headers. Will add more detailed documentation as I go.
2014-02-12 08:04:50 -07:00
|
|
|
OBS_DECLARE_MODULE()
|
2014-07-09 22:12:57 -07:00
|
|
|
OBS_MODULE_USE_DEFAULT_LOCALE("obs-ffmpeg", "en-US")
|
2014-01-19 03:16:41 -07:00
|
|
|
|
2015-03-04 12:45:50 -06:00
|
|
|
extern struct obs_source_info ffmpeg_source;
|
Implement RTMP module (still needs drop code)
- Implement the RTMP output module. This time around, we just use a
simple FLV muxer, then just write to the stream with RTMP_Write.
Easy and effective.
- Fix the FLV muxer, the muxer now outputs proper FLV packets.
- Output API:
* When using encoders, automatically interleave encoded packets
before sending it to the output.
* Pair encoders and have them automatically wait for the other to
start to ensure sync.
* Change 'obs_output_signal_start_fail' to 'obs_output_signal_stop'
because it was a bit confusing, and doing this makes a lot more
sense for outputs that need to stop suddenly (disconnections/etc).
- Encoder API:
* Remove some unnecessary encoder functions from the actual API and
make them internal. Most of the encoder functions are handled
automatically by outputs anyway, so there's no real need to expose
them and end up inadvertently confusing plugin writers.
* Have audio encoders wait for the video encoder to get a frame, then
start at the exact data point that the first video frame starts to
ensure the most accrate sync of video/audio possible.
* Add a required 'frame_size' callback for audio encoders that
returns the expected number of frames desired to encode with. This
way, the libobs encoder API can handle the circular buffering
internally automatically for the encoder modules, so encoder
writers don't have to do it themselves.
- Fix a few bugs in the serializer interface. It was passing the wrong
variable for the data in a few cases.
- If a source has video, make obs_source_update defer the actual update
callback until the tick function is called to prevent threading
issues.
2014-04-07 22:00:10 -07:00
|
|
|
extern struct obs_output_info ffmpeg_output;
|
2015-05-28 23:11:37 -07:00
|
|
|
extern struct obs_output_info ffmpeg_muxer;
|
Implement RTMP module (still needs drop code)
- Implement the RTMP output module. This time around, we just use a
simple FLV muxer, then just write to the stream with RTMP_Write.
Easy and effective.
- Fix the FLV muxer, the muxer now outputs proper FLV packets.
- Output API:
* When using encoders, automatically interleave encoded packets
before sending it to the output.
* Pair encoders and have them automatically wait for the other to
start to ensure sync.
* Change 'obs_output_signal_start_fail' to 'obs_output_signal_stop'
because it was a bit confusing, and doing this makes a lot more
sense for outputs that need to stop suddenly (disconnections/etc).
- Encoder API:
* Remove some unnecessary encoder functions from the actual API and
make them internal. Most of the encoder functions are handled
automatically by outputs anyway, so there's no real need to expose
them and end up inadvertently confusing plugin writers.
* Have audio encoders wait for the video encoder to get a frame, then
start at the exact data point that the first video frame starts to
ensure the most accrate sync of video/audio possible.
* Add a required 'frame_size' callback for audio encoders that
returns the expected number of frames desired to encode with. This
way, the libobs encoder API can handle the circular buffering
internally automatically for the encoder modules, so encoder
writers don't have to do it themselves.
- Fix a few bugs in the serializer interface. It was passing the wrong
variable for the data in a few cases.
- If a source has video, make obs_source_update defer the actual update
callback until the tick function is called to prevent threading
issues.
2014-04-07 22:00:10 -07:00
|
|
|
extern struct obs_encoder_info aac_encoder_info;
|
2014-01-19 03:16:41 -07:00
|
|
|
|
2015-08-13 12:55:21 -07:00
|
|
|
static DARRAY(struct log_context {
|
|
|
|
void *context;
|
|
|
|
char str[4096];
|
|
|
|
int print_prefix;
|
|
|
|
} *) active_log_contexts;
|
|
|
|
static DARRAY(struct log_context *) cached_log_contexts;
|
|
|
|
pthread_mutex_t log_contexts_mutex = PTHREAD_MUTEX_INITIALIZER;
|
|
|
|
|
|
|
|
static struct log_context *create_or_fetch_log_context(void *context)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&log_contexts_mutex);
|
|
|
|
for (size_t i = 0; i < active_log_contexts.num; i++) {
|
|
|
|
if (context == active_log_contexts.array[i]->context) {
|
|
|
|
pthread_mutex_unlock(&log_contexts_mutex);
|
|
|
|
return active_log_contexts.array[i];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct log_context *new_log_context = NULL;
|
|
|
|
|
|
|
|
size_t cnt = cached_log_contexts.num;
|
|
|
|
if (!!cnt) {
|
|
|
|
new_log_context = cached_log_contexts.array[cnt - 1];
|
|
|
|
da_pop_back(cached_log_contexts);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!new_log_context)
|
|
|
|
new_log_context = bzalloc(sizeof(struct log_context));
|
|
|
|
|
|
|
|
new_log_context->context = context;
|
|
|
|
new_log_context->str[0] = '\0';
|
|
|
|
new_log_context->print_prefix = 1;
|
|
|
|
|
|
|
|
da_push_back(active_log_contexts, &new_log_context);
|
|
|
|
|
2015-10-12 22:41:18 +02:00
|
|
|
pthread_mutex_unlock(&log_contexts_mutex);
|
|
|
|
|
2015-08-13 12:55:21 -07:00
|
|
|
return new_log_context;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void destroy_log_context(struct log_context *log_context)
|
|
|
|
{
|
|
|
|
pthread_mutex_lock(&log_contexts_mutex);
|
|
|
|
da_erase_item(active_log_contexts, &log_context);
|
|
|
|
da_push_back(cached_log_contexts, &log_context);
|
|
|
|
pthread_mutex_unlock(&log_contexts_mutex);
|
|
|
|
}
|
|
|
|
|
2015-08-04 08:52:12 -05:00
|
|
|
static void ffmpeg_log_callback(void* context, int level, const char* format,
|
|
|
|
va_list args)
|
|
|
|
{
|
|
|
|
if (format == NULL)
|
|
|
|
return;
|
|
|
|
|
2015-08-13 12:55:21 -07:00
|
|
|
struct log_context *log_context = create_or_fetch_log_context(context);
|
|
|
|
|
|
|
|
char *str = log_context->str;
|
2015-08-04 08:52:12 -05:00
|
|
|
|
|
|
|
av_log_format_line(context, level, format, args, str + strlen(str),
|
2015-08-13 12:55:21 -07:00
|
|
|
(int)(sizeof(log_context->str) - strlen(str)),
|
|
|
|
&log_context->print_prefix);
|
2015-08-04 08:52:12 -05:00
|
|
|
|
|
|
|
int obs_level;
|
|
|
|
switch (level) {
|
|
|
|
case AV_LOG_PANIC:
|
|
|
|
case AV_LOG_FATAL:
|
|
|
|
obs_level = LOG_ERROR;
|
|
|
|
break;
|
|
|
|
case AV_LOG_ERROR:
|
|
|
|
case AV_LOG_WARNING:
|
|
|
|
obs_level = LOG_WARNING;
|
|
|
|
break;
|
|
|
|
case AV_LOG_INFO:
|
|
|
|
case AV_LOG_VERBOSE:
|
|
|
|
obs_level = LOG_INFO;
|
|
|
|
break;
|
|
|
|
case AV_LOG_DEBUG:
|
|
|
|
default:
|
|
|
|
obs_level = LOG_DEBUG;
|
|
|
|
}
|
|
|
|
|
2015-08-13 12:55:21 -07:00
|
|
|
if (!log_context->print_prefix)
|
2015-08-04 08:52:12 -05:00
|
|
|
return;
|
|
|
|
|
|
|
|
char *str_end = str + strlen(str) - 1;
|
|
|
|
while(str < str_end) {
|
|
|
|
if (*str_end != '\n')
|
|
|
|
break;
|
|
|
|
*str_end-- = '\0';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (str_end <= str)
|
2015-08-13 12:55:21 -07:00
|
|
|
goto cleanup;
|
2015-08-04 08:52:12 -05:00
|
|
|
|
|
|
|
blog(obs_level, "[ffmpeg] %s", str);
|
2015-08-13 12:55:21 -07:00
|
|
|
|
|
|
|
cleanup:
|
|
|
|
destroy_log_context(log_context);
|
2015-08-04 08:52:12 -05:00
|
|
|
}
|
2015-03-04 12:45:50 -06:00
|
|
|
|
2014-07-27 12:42:43 -07:00
|
|
|
bool obs_module_load(void)
|
2014-02-07 03:03:54 -07:00
|
|
|
{
|
2015-08-13 12:55:21 -07:00
|
|
|
da_init(active_log_contexts);
|
|
|
|
da_init(cached_log_contexts);
|
|
|
|
|
2015-08-04 08:52:12 -05:00
|
|
|
av_log_set_callback(ffmpeg_log_callback);
|
2015-03-04 12:45:50 -06:00
|
|
|
|
|
|
|
obs_register_source(&ffmpeg_source);
|
Revamp API and start using doxygen
The API used to be designed in such a way to where it would expect
exports for each individual source/output/encoder/etc. You would export
functions for each and it would automatically load those functions based
on a specific naming scheme from the module.
The idea behind this was that I wanted to limit the usage of structures
in the API so only functions could be used. It was an interesting idea
in theory, but this idea turned out to be flawed in a number of ways:
1.) Requiring exports to create sources/outputs/encoders/etc meant that
you could not create them by any other means, which meant that
things like faruton's .net plugin would become difficult.
2.) Export function declarations could not be checked, therefore if you
created a function with the wrong parameters and parameter types,
the compiler wouldn't know how to check for that.
3.) Required overly complex load functions in libobs just to handle it.
It makes much more sense to just have a load function that you call
manually. Complexity is the bane of all good programs.
4.) It required that you have functions of specific names, which looked
and felt somewhat unsightly.
So, to fix these issues, I replaced it with a more commonly used API
scheme, seen commonly in places like kernels and typical C libraries
with abstraction. You simply create a structure that contains the
callback definitions, and you pass it to a function to register that
definition (such as obs_register_source), which you call in the
obs_module_load of the module.
It will also automatically check the structure size and ensure that it
only loads the required values if the structure happened to add new
values in an API change.
The "main" source file for each module must include obs-module.h, and
must use OBS_DECLARE_MODULE() within that source file.
Also, started writing some doxygen documentation in to the main library
headers. Will add more detailed documentation as I go.
2014-02-12 08:04:50 -07:00
|
|
|
obs_register_output(&ffmpeg_output);
|
2015-05-28 23:11:37 -07:00
|
|
|
obs_register_output(&ffmpeg_muxer);
|
Implement RTMP module (still needs drop code)
- Implement the RTMP output module. This time around, we just use a
simple FLV muxer, then just write to the stream with RTMP_Write.
Easy and effective.
- Fix the FLV muxer, the muxer now outputs proper FLV packets.
- Output API:
* When using encoders, automatically interleave encoded packets
before sending it to the output.
* Pair encoders and have them automatically wait for the other to
start to ensure sync.
* Change 'obs_output_signal_start_fail' to 'obs_output_signal_stop'
because it was a bit confusing, and doing this makes a lot more
sense for outputs that need to stop suddenly (disconnections/etc).
- Encoder API:
* Remove some unnecessary encoder functions from the actual API and
make them internal. Most of the encoder functions are handled
automatically by outputs anyway, so there's no real need to expose
them and end up inadvertently confusing plugin writers.
* Have audio encoders wait for the video encoder to get a frame, then
start at the exact data point that the first video frame starts to
ensure the most accrate sync of video/audio possible.
* Add a required 'frame_size' callback for audio encoders that
returns the expected number of frames desired to encode with. This
way, the libobs encoder API can handle the circular buffering
internally automatically for the encoder modules, so encoder
writers don't have to do it themselves.
- Fix a few bugs in the serializer interface. It was passing the wrong
variable for the data in a few cases.
- If a source has video, make obs_source_update defer the actual update
callback until the tick function is called to prevent threading
issues.
2014-04-07 22:00:10 -07:00
|
|
|
obs_register_encoder(&aac_encoder_info);
|
2014-02-07 03:03:54 -07:00
|
|
|
return true;
|
2014-01-19 03:16:41 -07:00
|
|
|
}
|
2015-08-13 12:55:21 -07:00
|
|
|
|
|
|
|
void obs_module_unload(void)
|
|
|
|
{
|
|
|
|
av_log_set_callback(av_log_default_callback);
|
|
|
|
|
|
|
|
#ifdef _WIN32
|
|
|
|
pthread_mutex_destroy(&log_contexts_mutex);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
for (size_t i = 0; i < active_log_contexts.num; i++) {
|
|
|
|
bfree(active_log_contexts.array[i]);
|
|
|
|
}
|
|
|
|
for (size_t i = 0; i < cached_log_contexts.num; i++) {
|
|
|
|
bfree(cached_log_contexts.array[i]);
|
|
|
|
}
|
|
|
|
|
|
|
|
da_free(active_log_contexts);
|
|
|
|
da_free(cached_log_contexts);
|
2015-10-12 22:41:18 +02:00
|
|
|
}
|