123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142 |
- #include <stdlib.h>
- #include <util/threading.h>
- #include <util/platform.h>
- #include <obs.h>
- struct async_sync_test {
- obs_source_t *source;
- os_event_t *stop_signal;
- pthread_t thread;
- bool initialized;
- };
- /* middle C */
- static const double rate = 261.63 / 48000.0;
- #ifndef M_PI
- #define M_PI 3.1415926535897932384626433832795
- #endif
- #define M_PI_X2 M_PI * 2
- static const char *ast_getname(void *unused)
- {
- UNUSED_PARAMETER(unused);
- return "Sync Test (Async Video/Audio Source)";
- }
- static void ast_destroy(void *data)
- {
- struct async_sync_test *ast = data;
- if (ast->initialized) {
- os_event_signal(ast->stop_signal);
- pthread_join(ast->thread, NULL);
- }
- os_event_destroy(ast->stop_signal);
- bfree(ast);
- }
- static inline void fill_texture(uint32_t *pixels, uint32_t color)
- {
- size_t x, y;
- for (y = 0; y < 20; y++) {
- for (x = 0; x < 20; x++) {
- pixels[y * 20 + x] = color;
- }
- }
- }
- static void *video_thread(void *data)
- {
- struct async_sync_test *ast = data;
- uint32_t sample_rate = audio_output_get_sample_rate(obs_get_audio());
- uint32_t *pixels = bmalloc(20 * 20 * sizeof(uint32_t));
- float *samples = bmalloc(sample_rate * sizeof(float));
- uint64_t cur_time = os_gettime_ns();
- bool whitelist = false;
- double cos_val = 0.0;
- uint64_t start_time = cur_time;
- struct obs_source_frame frame = {
- .data = {[0] = (uint8_t *)pixels},
- .linesize = {[0] = 20 * 4},
- .width = 20,
- .height = 20,
- .format = VIDEO_FORMAT_BGRX,
- };
- struct obs_source_audio audio = {
- .speakers = SPEAKERS_MONO,
- .data = {[0] = (uint8_t *)samples},
- .samples_per_sec = sample_rate,
- .frames = sample_rate,
- .format = AUDIO_FORMAT_FLOAT,
- };
- while (os_event_try(ast->stop_signal) == EAGAIN) {
- fill_texture(pixels, whitelist ? 0xFFFFFFFF : 0xFF000000);
- frame.timestamp = cur_time - start_time;
- audio.timestamp = cur_time - start_time;
- if (whitelist) {
- for (size_t i = 0; i < sample_rate; i++) {
- cos_val += rate * M_PI_X2;
- if (cos_val > M_PI_X2)
- cos_val -= M_PI_X2;
- samples[i] = (float)(cos(cos_val) * 0.5);
- }
- } else {
- for (size_t i = 0; i < sample_rate; i++)
- samples[i] = 0.0f;
- }
- obs_source_output_video(ast->source, &frame);
- obs_source_output_audio(ast->source, &audio);
- os_sleepto_ns(cur_time += 1000000000);
- whitelist = !whitelist;
- }
- bfree(pixels);
- bfree(samples);
- return NULL;
- }
- static void *ast_create(obs_data_t *settings, obs_source_t *source)
- {
- struct async_sync_test *ast = bzalloc(sizeof(struct async_sync_test));
- ast->source = source;
- if (os_event_init(&ast->stop_signal, OS_EVENT_TYPE_MANUAL) != 0) {
- ast_destroy(ast);
- return NULL;
- }
- if (pthread_create(&ast->thread, NULL, video_thread, ast) != 0) {
- ast_destroy(ast);
- return NULL;
- }
- ast->initialized = true;
- UNUSED_PARAMETER(settings);
- UNUSED_PARAMETER(source);
- return ast;
- }
- struct obs_source_info async_sync_test = {
- .id = "async_sync_test",
- .type = OBS_SOURCE_TYPE_INPUT,
- .output_flags = OBS_SOURCE_ASYNC_VIDEO | OBS_SOURCE_AUDIO,
- .get_name = ast_getname,
- .create = ast_create,
- .destroy = ast_destroy,
- };
|