async-delay-filter.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. #include <obs-module.h>
  2. #include <util/deque.h>
  3. #include <util/util_uint64.h>
  4. /* NOTE: Delaying audio shouldn't be necessary because the audio subsystem will
  5. * automatically sync audio to video frames */
  6. /* #define DELAY_AUDIO */
  7. #ifndef SEC_TO_NSEC
  8. #define SEC_TO_NSEC 1000000000ULL
  9. #endif
  10. #ifndef MSEC_TO_NSEC
  11. #define MSEC_TO_NSEC 1000000ULL
  12. #endif
  13. #define SETTING_DELAY_MS "delay_ms"
  14. #define TEXT_DELAY_MS obs_module_text("DelayMs")
  15. struct async_delay_data {
  16. obs_source_t *context;
  17. /* contains struct obs_source_frame* */
  18. struct deque video_frames;
  19. #ifdef DELAY_AUDIO
  20. /* stores the audio data */
  21. struct deque audio_frames;
  22. struct obs_audio_data audio_output;
  23. #endif
  24. uint64_t last_video_ts;
  25. uint64_t last_audio_ts;
  26. uint64_t interval;
  27. uint64_t samplerate;
  28. bool video_delay_reached;
  29. bool audio_delay_reached;
  30. bool reset_video;
  31. bool reset_audio;
  32. };
  33. static const char *async_delay_filter_name(void *unused)
  34. {
  35. UNUSED_PARAMETER(unused);
  36. return obs_module_text("AsyncDelayFilter");
  37. }
  38. static void free_video_data(struct async_delay_data *filter, obs_source_t *parent)
  39. {
  40. while (filter->video_frames.size) {
  41. struct obs_source_frame *frame;
  42. deque_pop_front(&filter->video_frames, &frame, sizeof(struct obs_source_frame *));
  43. obs_source_release_frame(parent, frame);
  44. }
  45. }
  46. #ifdef DELAY_AUDIO
  47. static inline void free_audio_packet(struct obs_audio_data *audio)
  48. {
  49. for (size_t i = 0; i < MAX_AV_PLANES; i++)
  50. bfree(audio->data[i]);
  51. memset(audio, 0, sizeof(*audio));
  52. }
  53. static void free_audio_data(struct async_delay_data *filter)
  54. {
  55. while (filter->audio_frames.size) {
  56. struct obs_audio_data audio;
  57. deque_pop_front(&filter->audio_frames, &audio, sizeof(struct obs_audio_data));
  58. free_audio_packet(&audio);
  59. }
  60. }
  61. #endif
  62. static void async_delay_filter_update(void *data, obs_data_t *settings)
  63. {
  64. struct async_delay_data *filter = data;
  65. uint64_t new_interval = (uint64_t)obs_data_get_int(settings, SETTING_DELAY_MS) * MSEC_TO_NSEC;
  66. if (new_interval < filter->interval)
  67. free_video_data(filter, obs_filter_get_parent(filter->context));
  68. filter->reset_audio = true;
  69. filter->reset_video = true;
  70. filter->interval = new_interval;
  71. filter->video_delay_reached = false;
  72. filter->audio_delay_reached = false;
  73. }
  74. static void *async_delay_filter_create(obs_data_t *settings, obs_source_t *context)
  75. {
  76. struct async_delay_data *filter = bzalloc(sizeof(*filter));
  77. struct obs_audio_info oai;
  78. filter->context = context;
  79. async_delay_filter_update(filter, settings);
  80. obs_get_audio_info(&oai);
  81. filter->samplerate = oai.samples_per_sec;
  82. return filter;
  83. }
  84. static void async_delay_filter_destroy(void *data)
  85. {
  86. struct async_delay_data *filter = data;
  87. deque_free(&filter->video_frames);
  88. #ifdef DELAY_AUDIO
  89. free_audio_packet(&filter->audio_output);
  90. deque_free(&filter->audio_frames);
  91. #endif
  92. bfree(data);
  93. }
  94. static obs_properties_t *async_delay_filter_properties(void *data)
  95. {
  96. obs_properties_t *props = obs_properties_create();
  97. obs_property_t *p = obs_properties_add_int(props, SETTING_DELAY_MS, TEXT_DELAY_MS, 0, 20000, 1);
  98. obs_property_int_set_suffix(p, " ms");
  99. UNUSED_PARAMETER(data);
  100. return props;
  101. }
  102. static void async_delay_filter_remove(void *data, obs_source_t *parent)
  103. {
  104. struct async_delay_data *filter = data;
  105. free_video_data(filter, parent);
  106. #ifdef DELAY_AUDIO
  107. free_audio_data(filter);
  108. #endif
  109. }
  110. /* due to the fact that we need timing information to be consistent in order to
  111. * measure the current interval of data, if there is an unexpected hiccup or
  112. * jump with the timestamps, reset the cached delay data and start again to
  113. * ensure that the timing is consistent */
  114. static inline bool is_timestamp_jump(uint64_t ts, uint64_t prev_ts)
  115. {
  116. return ts < prev_ts || (ts - prev_ts) > SEC_TO_NSEC;
  117. }
  118. static struct obs_source_frame *async_delay_filter_video(void *data, struct obs_source_frame *frame)
  119. {
  120. struct async_delay_data *filter = data;
  121. obs_source_t *parent = obs_filter_get_parent(filter->context);
  122. struct obs_source_frame *output;
  123. uint64_t cur_interval;
  124. if (filter->reset_video || is_timestamp_jump(frame->timestamp, filter->last_video_ts)) {
  125. free_video_data(filter, parent);
  126. filter->video_delay_reached = false;
  127. filter->reset_video = false;
  128. }
  129. filter->last_video_ts = frame->timestamp;
  130. deque_push_back(&filter->video_frames, &frame, sizeof(struct obs_source_frame *));
  131. deque_peek_front(&filter->video_frames, &output, sizeof(struct obs_source_frame *));
  132. cur_interval = frame->timestamp - output->timestamp;
  133. if (!filter->video_delay_reached && cur_interval < filter->interval)
  134. return NULL;
  135. deque_pop_front(&filter->video_frames, NULL, sizeof(struct obs_source_frame *));
  136. if (!filter->video_delay_reached)
  137. filter->video_delay_reached = true;
  138. return output;
  139. }
  140. #ifdef DELAY_AUDIO
  141. static struct obs_audio_data *async_delay_filter_audio(void *data, struct obs_audio_data *audio)
  142. {
  143. struct async_delay_data *filter = data;
  144. struct obs_audio_data cached = *audio;
  145. uint64_t cur_interval;
  146. uint64_t duration;
  147. uint64_t end_ts;
  148. if (filter->reset_audio || is_timestamp_jump(audio->timestamp, filter->last_audio_ts)) {
  149. free_audio_data(filter);
  150. filter->audio_delay_reached = false;
  151. filter->reset_audio = false;
  152. }
  153. filter->last_audio_ts = audio->timestamp;
  154. duration = util_mul_div64(audio->frames, SEC_TO_NSEC, filter->samplerate);
  155. end_ts = audio->timestamp + duration;
  156. for (size_t i = 0; i < MAX_AV_PLANES; i++) {
  157. if (!audio->data[i])
  158. break;
  159. cached.data[i] = bmemdup(audio->data[i], audio->frames * sizeof(float));
  160. }
  161. free_audio_packet(&filter->audio_output);
  162. deque_push_back(&filter->audio_frames, &cached, sizeof(cached));
  163. deque_peek_front(&filter->audio_frames, &cached, sizeof(cached));
  164. cur_interval = end_ts - cached.timestamp;
  165. if (!filter->audio_delay_reached && cur_interval < filter->interval)
  166. return NULL;
  167. deque_pop_front(&filter->audio_frames, NULL, sizeof(cached));
  168. memcpy(&filter->audio_output, &cached, sizeof(cached));
  169. if (!filter->audio_delay_reached)
  170. filter->audio_delay_reached = true;
  171. return &filter->audio_output;
  172. }
  173. #endif
  174. struct obs_source_info async_delay_filter = {
  175. .id = "async_delay_filter",
  176. .type = OBS_SOURCE_TYPE_FILTER,
  177. .output_flags = OBS_SOURCE_VIDEO | OBS_SOURCE_ASYNC,
  178. .get_name = async_delay_filter_name,
  179. .create = async_delay_filter_create,
  180. .destroy = async_delay_filter_destroy,
  181. .update = async_delay_filter_update,
  182. .get_properties = async_delay_filter_properties,
  183. .filter_video = async_delay_filter_video,
  184. #ifdef DELAY_AUDIO
  185. .filter_audio = async_delay_filter_audio,
  186. #endif
  187. .filter_remove = async_delay_filter_remove,
  188. };