123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193 |
- #include <stdlib.h>
- #include <util/threading.h>
- #include <util/platform.h>
- #include <obs.h>
- struct buffering_async_sync_test {
- obs_source_t *source;
- os_event_t *stop_signal;
- pthread_t thread;
- bool initialized;
- bool buffer_audio;
- };
- #define CYCLE_COUNT 7
- static const double aud_rates[CYCLE_COUNT] = {
- 220.00 / 48000.0, /* A */
- 233.08 / 48000.0, /* A# */
- 246.94 / 48000.0, /* B */
- 261.63 / 48000.0, /* C */
- 277.18 / 48000.0, /* C# */
- 293.67 / 48000.0, /* D */
- 311.13 / 48000.0, /* D# */
- };
- #define MAKE_COL_CHAN(x, y) (((0xFF / 7) * x) << (y * 8))
- #define MAKE_GRAYSCALE(x) (MAKE_COL_CHAN(x, 0) | MAKE_COL_CHAN(x, 1) | MAKE_COL_CHAN(x, 2))
- static const uint32_t vid_colors[CYCLE_COUNT] = {
- 0xFF000000,
- 0xFF000000 + MAKE_GRAYSCALE(1),
- 0xFF000000 + MAKE_GRAYSCALE(2),
- 0xFF000000 + MAKE_GRAYSCALE(3),
- 0xFF000000 + MAKE_GRAYSCALE(4),
- 0xFF000000 + MAKE_GRAYSCALE(5),
- 0xFF000000 + MAKE_GRAYSCALE(6),
- };
- #ifndef M_PI
- #define M_PI 3.1415926535897932384626433832795
- #endif
- #define M_PI_X2 M_PI * 2
- static const char *bast_getname(void *unused)
- {
- UNUSED_PARAMETER(unused);
- return "Audio Buffering Sync Test (Async Video/Audio Source)";
- }
- static void bast_destroy(void *data)
- {
- struct buffering_async_sync_test *bast = data;
- if (bast->initialized) {
- os_event_signal(bast->stop_signal);
- pthread_join(bast->thread, NULL);
- }
- os_event_destroy(bast->stop_signal);
- bfree(bast);
- }
- 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 buffering_async_sync_test *bast = 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();
- int cur_vid_pos = 0;
- int cur_aud_pos = 0;
- double cos_val = 0.0;
- uint64_t start_time = cur_time;
- bool audio_buffering_enabled = false;
- 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 / 4,
- .format = AUDIO_FORMAT_FLOAT,
- };
- while (os_event_try(bast->stop_signal) == EAGAIN) {
- fill_texture(pixels, vid_colors[cur_vid_pos]);
- if (!audio_buffering_enabled && bast->buffer_audio) {
- audio_buffering_enabled = true;
- blog(LOG_DEBUG, "okay, buffering audio: now");
- /* 1 second = 4 cycles when running at
- * 250ms per cycle */
- cur_aud_pos -= 4;
- if (cur_aud_pos < 0)
- cur_aud_pos += CYCLE_COUNT;
- }
- /* should cause approximately 750 milliseconds of audio
- * buffering */
- frame.timestamp = cur_time - start_time;
- audio.timestamp = cur_time - start_time - (audio_buffering_enabled ? 1000000000 : 0);
- const double rate = aud_rates[cur_aud_pos];
- for (size_t i = 0; i < sample_rate / 4; 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);
- }
- obs_source_output_video(bast->source, &frame);
- obs_source_output_audio(bast->source, &audio);
- os_sleepto_ns(cur_time += 250000000);
- if (++cur_vid_pos == CYCLE_COUNT)
- cur_vid_pos = 0;
- if (++cur_aud_pos == CYCLE_COUNT)
- cur_aud_pos = 0;
- }
- bfree(pixels);
- bfree(samples);
- return NULL;
- }
- static void bast_buffer_audio(void *data, obs_hotkey_id id, obs_hotkey_t *hotkey, bool pressed)
- {
- struct buffering_async_sync_test *bast = data;
- UNUSED_PARAMETER(id);
- UNUSED_PARAMETER(hotkey);
- if (pressed)
- bast->buffer_audio = true;
- }
- static void *bast_create(obs_data_t *settings, obs_source_t *source)
- {
- struct buffering_async_sync_test *bast = bzalloc(sizeof(*bast));
- bast->source = source;
- if (os_event_init(&bast->stop_signal, OS_EVENT_TYPE_MANUAL) != 0) {
- bast_destroy(bast);
- return NULL;
- }
- if (pthread_create(&bast->thread, NULL, video_thread, bast) != 0) {
- bast_destroy(bast);
- return NULL;
- }
- obs_hotkey_register_source(source, "AudioBufferingSyncTest.Buffer", "Buffer Audio", bast_buffer_audio, bast);
- bast->initialized = true;
- UNUSED_PARAMETER(settings);
- UNUSED_PARAMETER(source);
- return bast;
- }
- struct obs_source_info buffering_async_sync_test = {
- .id = "buffering_async_sync_test",
- .type = OBS_SOURCE_TYPE_INPUT,
- .output_flags = OBS_SOURCE_ASYNC_VIDEO | OBS_SOURCE_AUDIO,
- .get_name = bast_getname,
- .create = bast_create,
- .destroy = bast_destroy,
- };
|