obs-output-delay.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. /******************************************************************************
  2. Copyright (C) 2023 by Lain Bailey <[email protected]>
  3. This program is free software: you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation, either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program. If not, see <http://www.gnu.org/licenses/>.
  13. ******************************************************************************/
  14. #include <inttypes.h>
  15. #include "obs-internal.h"
  16. static inline bool delay_active(const struct obs_output *output)
  17. {
  18. return os_atomic_load_bool(&output->delay_active);
  19. }
  20. static inline bool delay_capturing(const struct obs_output *output)
  21. {
  22. return os_atomic_load_bool(&output->delay_capturing);
  23. }
  24. static inline bool flag_encoded(const struct obs_output *output)
  25. {
  26. return (output->info.flags & OBS_OUTPUT_ENCODED) != 0;
  27. }
  28. static inline bool log_flag_encoded(const struct obs_output *output, const char *func_name, bool inverse_log)
  29. {
  30. const char *prefix = inverse_log ? "n encoded" : " raw";
  31. bool ret = flag_encoded(output);
  32. if ((!inverse_log && !ret) || (inverse_log && ret))
  33. blog(LOG_WARNING, "Output '%s': Tried to use %s on a%s output", output->context.name, func_name,
  34. prefix);
  35. return ret;
  36. }
  37. static inline void push_packet(struct obs_output *output, struct encoder_packet *packet,
  38. struct encoder_packet_time *packet_time, uint64_t t)
  39. {
  40. struct delay_data dd;
  41. dd.msg = DELAY_MSG_PACKET;
  42. dd.ts = t;
  43. dd.packet_time_valid = packet_time != NULL;
  44. if (packet_time != NULL)
  45. dd.packet_time = *packet_time;
  46. obs_encoder_packet_create_instance(&dd.packet, packet);
  47. pthread_mutex_lock(&output->delay_mutex);
  48. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  49. pthread_mutex_unlock(&output->delay_mutex);
  50. }
  51. static inline void process_delay_data(struct obs_output *output, struct delay_data *dd)
  52. {
  53. switch (dd->msg) {
  54. case DELAY_MSG_PACKET:
  55. if (!delay_active(output) || !delay_capturing(output))
  56. obs_encoder_packet_release(&dd->packet);
  57. else
  58. output->delay_callback(output, &dd->packet, dd->packet_time_valid ? &dd->packet_time : NULL);
  59. break;
  60. case DELAY_MSG_START:
  61. obs_output_actual_start(output);
  62. break;
  63. case DELAY_MSG_STOP:
  64. obs_output_actual_stop(output, false, dd->ts);
  65. break;
  66. }
  67. }
  68. void obs_output_cleanup_delay(obs_output_t *output)
  69. {
  70. struct delay_data dd;
  71. while (output->delay_data.size) {
  72. deque_pop_front(&output->delay_data, &dd, sizeof(dd));
  73. if (dd.msg == DELAY_MSG_PACKET) {
  74. obs_encoder_packet_release(&dd.packet);
  75. }
  76. }
  77. output->active_delay_ns = 0;
  78. os_atomic_set_long(&output->delay_restart_refs, 0);
  79. }
  80. static inline bool pop_packet(struct obs_output *output, uint64_t t)
  81. {
  82. uint64_t elapsed_time;
  83. struct delay_data dd;
  84. bool popped = false;
  85. bool preserve;
  86. /* ------------------------------------------------ */
  87. preserve = (output->delay_cur_flags & OBS_OUTPUT_DELAY_PRESERVE) != 0;
  88. pthread_mutex_lock(&output->delay_mutex);
  89. if (output->delay_data.size) {
  90. deque_peek_front(&output->delay_data, &dd, sizeof(dd));
  91. elapsed_time = (t - dd.ts);
  92. if (preserve && output->reconnecting) {
  93. output->active_delay_ns = elapsed_time;
  94. } else if (elapsed_time > output->active_delay_ns) {
  95. deque_pop_front(&output->delay_data, NULL, sizeof(dd));
  96. popped = true;
  97. }
  98. }
  99. pthread_mutex_unlock(&output->delay_mutex);
  100. /* ------------------------------------------------ */
  101. if (popped)
  102. process_delay_data(output, &dd);
  103. return popped;
  104. }
  105. void process_delay(void *data, struct encoder_packet *packet, struct encoder_packet_time *packet_time)
  106. {
  107. struct obs_output *output = data;
  108. uint64_t t = os_gettime_ns();
  109. push_packet(output, packet, packet_time, t);
  110. while (pop_packet(output, t))
  111. ;
  112. }
  113. bool obs_output_delay_start(obs_output_t *output)
  114. {
  115. struct delay_data dd = {
  116. .msg = DELAY_MSG_START,
  117. .ts = os_gettime_ns(),
  118. };
  119. if (!delay_active(output)) {
  120. bool can_begin = obs_output_can_begin_data_capture(output, 0);
  121. if (!can_begin)
  122. return false;
  123. if (!obs_output_initialize_encoders(output, 0))
  124. return false;
  125. }
  126. pthread_mutex_lock(&output->delay_mutex);
  127. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  128. pthread_mutex_unlock(&output->delay_mutex);
  129. os_atomic_inc_long(&output->delay_restart_refs);
  130. if (delay_active(output)) {
  131. do_output_signal(output, "starting");
  132. return true;
  133. }
  134. if (!obs_output_begin_data_capture(output, 0)) {
  135. obs_output_cleanup_delay(output);
  136. return false;
  137. }
  138. return true;
  139. }
  140. void obs_output_delay_stop(obs_output_t *output)
  141. {
  142. struct delay_data dd = {
  143. .msg = DELAY_MSG_STOP,
  144. .ts = os_gettime_ns(),
  145. };
  146. pthread_mutex_lock(&output->delay_mutex);
  147. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  148. pthread_mutex_unlock(&output->delay_mutex);
  149. do_output_signal(output, "stopping");
  150. }
  151. void obs_output_set_delay(obs_output_t *output, uint32_t delay_sec, uint32_t flags)
  152. {
  153. if (!obs_output_valid(output, "obs_output_set_delay"))
  154. return;
  155. if (!log_flag_encoded(output, __FUNCTION__, false))
  156. return;
  157. output->delay_sec = delay_sec;
  158. output->delay_flags = flags;
  159. }
  160. uint32_t obs_output_get_delay(const obs_output_t *output)
  161. {
  162. return obs_output_valid(output, "obs_output_set_delay") ? output->delay_sec : 0;
  163. }
  164. uint32_t obs_output_get_active_delay(const obs_output_t *output)
  165. {
  166. return obs_output_valid(output, "obs_output_set_delay") ? (uint32_t)(output->active_delay_ns / 1000000000ULL)
  167. : 0;
  168. }