obs-output-delay.c 5.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223
  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,
  29. const char *func_name, bool inverse_log)
  30. {
  31. const char *prefix = inverse_log ? "n encoded" : " raw";
  32. bool ret = flag_encoded(output);
  33. if ((!inverse_log && !ret) || (inverse_log && ret))
  34. blog(LOG_WARNING, "Output '%s': Tried to use %s on a%s output",
  35. output->context.name, func_name, prefix);
  36. return ret;
  37. }
  38. static inline void push_packet(struct obs_output *output,
  39. struct encoder_packet *packet, uint64_t t)
  40. {
  41. struct delay_data dd;
  42. dd.msg = DELAY_MSG_PACKET;
  43. dd.ts = t;
  44. obs_encoder_packet_create_instance(&dd.packet, packet);
  45. pthread_mutex_lock(&output->delay_mutex);
  46. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  47. pthread_mutex_unlock(&output->delay_mutex);
  48. }
  49. static inline void process_delay_data(struct obs_output *output,
  50. struct delay_data *dd)
  51. {
  52. switch (dd->msg) {
  53. case DELAY_MSG_PACKET:
  54. if (!delay_active(output) || !delay_capturing(output))
  55. obs_encoder_packet_release(&dd->packet);
  56. else
  57. output->delay_callback(output, &dd->packet);
  58. break;
  59. case DELAY_MSG_START:
  60. obs_output_actual_start(output);
  61. break;
  62. case DELAY_MSG_STOP:
  63. obs_output_actual_stop(output, false, dd->ts);
  64. break;
  65. }
  66. }
  67. void obs_output_cleanup_delay(obs_output_t *output)
  68. {
  69. struct delay_data dd;
  70. while (output->delay_data.size) {
  71. deque_pop_front(&output->delay_data, &dd, sizeof(dd));
  72. if (dd.msg == DELAY_MSG_PACKET) {
  73. obs_encoder_packet_release(&dd.packet);
  74. }
  75. }
  76. output->active_delay_ns = 0;
  77. os_atomic_set_long(&output->delay_restart_refs, 0);
  78. }
  79. static inline bool pop_packet(struct obs_output *output, uint64_t t)
  80. {
  81. uint64_t elapsed_time;
  82. struct delay_data dd;
  83. bool popped = false;
  84. bool preserve;
  85. /* ------------------------------------------------ */
  86. preserve = (output->delay_cur_flags & OBS_OUTPUT_DELAY_PRESERVE) != 0;
  87. pthread_mutex_lock(&output->delay_mutex);
  88. if (output->delay_data.size) {
  89. deque_peek_front(&output->delay_data, &dd, sizeof(dd));
  90. elapsed_time = (t - dd.ts);
  91. if (preserve && output->reconnecting) {
  92. output->active_delay_ns = elapsed_time;
  93. } else if (elapsed_time > output->active_delay_ns) {
  94. deque_pop_front(&output->delay_data, NULL, sizeof(dd));
  95. popped = true;
  96. }
  97. }
  98. pthread_mutex_unlock(&output->delay_mutex);
  99. /* ------------------------------------------------ */
  100. if (popped)
  101. process_delay_data(output, &dd);
  102. return popped;
  103. }
  104. void process_delay(void *data, struct encoder_packet *packet)
  105. {
  106. struct obs_output *output = data;
  107. uint64_t t = os_gettime_ns();
  108. push_packet(output, packet, t);
  109. while (pop_packet(output, t))
  110. ;
  111. }
  112. void obs_output_signal_delay(obs_output_t *output, const char *signal)
  113. {
  114. struct calldata params;
  115. uint8_t stack[128];
  116. calldata_init_fixed(&params, stack, sizeof(stack));
  117. calldata_set_ptr(&params, "output", output);
  118. calldata_set_int(&params, "sec", output->active_delay_ns / 1000000000);
  119. signal_handler_signal(output->context.signals, signal, &params);
  120. }
  121. bool obs_output_delay_start(obs_output_t *output)
  122. {
  123. struct delay_data dd = {
  124. .msg = DELAY_MSG_START,
  125. .ts = os_gettime_ns(),
  126. };
  127. if (!delay_active(output)) {
  128. bool can_begin = obs_output_can_begin_data_capture(output, 0);
  129. if (!can_begin)
  130. return false;
  131. if (!obs_output_initialize_encoders(output, 0))
  132. return false;
  133. }
  134. pthread_mutex_lock(&output->delay_mutex);
  135. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  136. pthread_mutex_unlock(&output->delay_mutex);
  137. os_atomic_inc_long(&output->delay_restart_refs);
  138. if (delay_active(output)) {
  139. do_output_signal(output, "starting");
  140. return true;
  141. }
  142. if (!obs_output_begin_data_capture(output, 0)) {
  143. obs_output_cleanup_delay(output);
  144. return false;
  145. }
  146. return true;
  147. }
  148. void obs_output_delay_stop(obs_output_t *output)
  149. {
  150. struct delay_data dd = {
  151. .msg = DELAY_MSG_STOP,
  152. .ts = os_gettime_ns(),
  153. };
  154. pthread_mutex_lock(&output->delay_mutex);
  155. deque_push_back(&output->delay_data, &dd, sizeof(dd));
  156. pthread_mutex_unlock(&output->delay_mutex);
  157. do_output_signal(output, "stopping");
  158. }
  159. void obs_output_set_delay(obs_output_t *output, uint32_t delay_sec,
  160. uint32_t flags)
  161. {
  162. if (!obs_output_valid(output, "obs_output_set_delay"))
  163. return;
  164. if (!log_flag_encoded(output, __FUNCTION__, false))
  165. return;
  166. output->delay_sec = delay_sec;
  167. output->delay_flags = flags;
  168. }
  169. uint32_t obs_output_get_delay(const obs_output_t *output)
  170. {
  171. return obs_output_valid(output, "obs_output_set_delay")
  172. ? output->delay_sec
  173. : 0;
  174. }
  175. uint32_t obs_output_get_active_delay(const obs_output_t *output)
  176. {
  177. return obs_output_valid(output, "obs_output_set_delay")
  178. ? (uint32_t)(output->active_delay_ns / 1000000000ULL)
  179. : 0;
  180. }