obs-source-deinterlace.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. /******************************************************************************
  2. Copyright (C) 2016 by Hugh 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 "obs-internal.h"
  15. static bool ready_deinterlace_frames(obs_source_t *source, uint64_t sys_time)
  16. {
  17. struct obs_source_frame *next_frame = source->async_frames.array[0];
  18. struct obs_source_frame *prev_frame = NULL;
  19. struct obs_source_frame *frame = NULL;
  20. uint64_t sys_offset = sys_time - source->last_sys_timestamp;
  21. uint64_t frame_time = next_frame->timestamp;
  22. uint64_t frame_offset = 0;
  23. size_t idx = 1;
  24. if (source->async_unbuffered) {
  25. while (source->async_frames.num > 2) {
  26. da_erase(source->async_frames, 0);
  27. remove_async_frame(source, next_frame);
  28. next_frame = source->async_frames.array[0];
  29. }
  30. if (source->async_frames.num == 2)
  31. source->async_frames.array[0]->prev_frame = true;
  32. source->deinterlace_offset = 0;
  33. source->last_frame_ts = next_frame->timestamp;
  34. return true;
  35. }
  36. /* account for timestamp invalidation */
  37. if (frame_out_of_bounds(source, frame_time)) {
  38. source->last_frame_ts = next_frame->timestamp;
  39. source->deinterlace_offset = 0;
  40. return true;
  41. } else {
  42. frame_offset = frame_time - source->last_frame_ts;
  43. source->last_frame_ts += sys_offset;
  44. }
  45. while (source->last_frame_ts > next_frame->timestamp) {
  46. /* this tries to reduce the needless frame duplication, also
  47. * helps smooth out async rendering to frame boundaries. In
  48. * other words, tries to keep the framerate as smooth as
  49. * possible */
  50. if ((source->last_frame_ts - next_frame->timestamp) < 2000000)
  51. break;
  52. if (prev_frame) {
  53. da_erase(source->async_frames, 0);
  54. remove_async_frame(source, prev_frame);
  55. }
  56. if (source->async_frames.num <= 2) {
  57. bool exit = true;
  58. if (prev_frame) {
  59. prev_frame->prev_frame = true;
  60. } else if (!frame && source->async_frames.num == 2) {
  61. exit = false;
  62. }
  63. if (exit) {
  64. source->deinterlace_offset = 0;
  65. return true;
  66. }
  67. }
  68. if (frame)
  69. idx = 2;
  70. else
  71. idx = 1;
  72. prev_frame = frame;
  73. frame = next_frame;
  74. next_frame = source->async_frames.array[idx];
  75. /* more timestamp checking and compensating */
  76. if ((next_frame->timestamp - frame_time) > MAX_TS_VAR) {
  77. source->last_frame_ts =
  78. next_frame->timestamp - frame_offset;
  79. source->deinterlace_offset = 0;
  80. }
  81. frame_time = next_frame->timestamp;
  82. frame_offset = frame_time - source->last_frame_ts;
  83. }
  84. if (prev_frame)
  85. prev_frame->prev_frame = true;
  86. return frame != NULL;
  87. }
  88. static inline bool first_frame(obs_source_t *s)
  89. {
  90. if (s->last_frame_ts)
  91. return false;
  92. if (s->async_frames.num >= 2)
  93. s->async_frames.array[0]->prev_frame = true;
  94. return true;
  95. }
  96. static inline uint64_t uint64_diff(uint64_t ts1, uint64_t ts2)
  97. {
  98. return (ts1 < ts2) ? (ts2 - ts1) : (ts1 - ts2);
  99. }
  100. static inline void deinterlace_get_closest_frames(obs_source_t *s,
  101. uint64_t sys_time)
  102. {
  103. const struct video_output_info *info;
  104. uint64_t half_interval;
  105. if (!s->async_frames.num)
  106. return;
  107. info = video_output_get_info(obs->video.video);
  108. half_interval = (uint64_t)info->fps_den * 500000000ULL /
  109. (uint64_t)info->fps_num;
  110. if (first_frame(s) || ready_deinterlace_frames(s, sys_time)) {
  111. uint64_t offset;
  112. s->prev_async_frame = NULL;
  113. s->cur_async_frame = s->async_frames.array[0];
  114. da_erase(s->async_frames, 0);
  115. if (s->cur_async_frame->prev_frame) {
  116. s->prev_async_frame = s->cur_async_frame;
  117. s->cur_async_frame = s->async_frames.array[0];
  118. da_erase(s->async_frames, 0);
  119. s->deinterlace_half_duration = (uint32_t)
  120. ((s->cur_async_frame->timestamp -
  121. s->prev_async_frame->timestamp) / 2);
  122. } else {
  123. s->deinterlace_half_duration = (uint32_t)
  124. ((s->cur_async_frame->timestamp -
  125. s->deinterlace_frame_ts) / 2);
  126. }
  127. if (!s->last_frame_ts)
  128. s->last_frame_ts = s->cur_async_frame->timestamp;
  129. s->deinterlace_frame_ts = s->cur_async_frame->timestamp;
  130. offset = obs->video.video_time - s->deinterlace_frame_ts;
  131. if (!s->deinterlace_offset) {
  132. s->deinterlace_offset = offset;
  133. } else {
  134. uint64_t offset_diff = uint64_diff(
  135. s->deinterlace_offset, offset);
  136. if (offset_diff > half_interval)
  137. s->deinterlace_offset = offset;
  138. }
  139. }
  140. }
  141. void deinterlace_process_last_frame(obs_source_t *s, uint64_t sys_time)
  142. {
  143. if (s->prev_async_frame) {
  144. remove_async_frame(s, s->prev_async_frame);
  145. s->prev_async_frame = NULL;
  146. }
  147. if (s->cur_async_frame) {
  148. remove_async_frame(s, s->cur_async_frame);
  149. s->cur_async_frame = NULL;
  150. }
  151. deinterlace_get_closest_frames(s, sys_time);
  152. }
  153. void set_deinterlace_texture_size(obs_source_t *source)
  154. {
  155. if (source->async_gpu_conversion) {
  156. source->async_prev_texrender =
  157. gs_texrender_create(GS_BGRX, GS_ZS_NONE);
  158. source->async_prev_texture = gs_texture_create(
  159. source->async_convert_width,
  160. source->async_convert_height,
  161. source->async_texture_format,
  162. 1, NULL, GS_DYNAMIC);
  163. } else {
  164. enum gs_color_format format = convert_video_format(
  165. source->async_format);
  166. source->async_prev_texture = gs_texture_create(
  167. source->async_width, source->async_height,
  168. format, 1, NULL, GS_DYNAMIC);
  169. }
  170. }
  171. static inline struct obs_source_frame *get_prev_frame(obs_source_t *source,
  172. bool *updated)
  173. {
  174. struct obs_source_frame *frame = NULL;
  175. pthread_mutex_lock(&source->async_mutex);
  176. *updated = source->cur_async_frame != NULL;
  177. frame = source->prev_async_frame;
  178. source->prev_async_frame = NULL;
  179. if (frame)
  180. os_atomic_inc_long(&frame->refs);
  181. pthread_mutex_unlock(&source->async_mutex);
  182. return frame;
  183. }
  184. void deinterlace_update_async_video(obs_source_t *source)
  185. {
  186. struct obs_source_frame *frame;
  187. bool updated;
  188. if (source->deinterlace_rendered)
  189. return;
  190. frame = get_prev_frame(source, &updated);
  191. source->deinterlace_rendered = true;
  192. if (frame)
  193. frame = filter_async_video(source, frame);
  194. if (frame) {
  195. if (set_async_texture_size(source, frame)) {
  196. update_async_texture(source, frame,
  197. source->async_prev_texture,
  198. source->async_prev_texrender);
  199. }
  200. obs_source_release_frame(source, frame);
  201. } else if (updated) { /* swap cur/prev if no previous texture */
  202. gs_texture_t *prev_tex = source->async_prev_texture;
  203. source->async_prev_texture = source->async_texture;
  204. source->async_texture = prev_tex;
  205. if (source->async_texrender) {
  206. gs_texrender_t *prev = source->async_prev_texrender;
  207. source->async_prev_texrender = source->async_texrender;
  208. source->async_texrender = prev;
  209. }
  210. }
  211. }
  212. static inline gs_effect_t *get_effect(enum obs_deinterlace_mode mode)
  213. {
  214. switch (mode) {
  215. case OBS_DEINTERLACE_MODE_DISABLE: return NULL;
  216. case OBS_DEINTERLACE_MODE_DISCARD:
  217. return obs_load_effect(&obs->video.deinterlace_discard_effect,
  218. "deinterlace_discard.effect");
  219. case OBS_DEINTERLACE_MODE_RETRO:
  220. return obs_load_effect(&obs->video.deinterlace_discard_2x_effect,
  221. "deinterlace_discard_2x.effect");
  222. case OBS_DEINTERLACE_MODE_BLEND:
  223. return obs_load_effect(&obs->video.deinterlace_blend_effect,
  224. "deinterlace_blend.effect");
  225. case OBS_DEINTERLACE_MODE_BLEND_2X:
  226. return obs_load_effect(&obs->video.deinterlace_blend_2x_effect,
  227. "deinterlace_blend_2x.effect");
  228. case OBS_DEINTERLACE_MODE_LINEAR:
  229. return obs_load_effect(&obs->video.deinterlace_linear_effect,
  230. "deinterlace_linear.effect");
  231. case OBS_DEINTERLACE_MODE_LINEAR_2X:
  232. return obs_load_effect(&obs->video.deinterlace_linear_2x_effect,
  233. "deinterlace_linear_2x.effect");
  234. case OBS_DEINTERLACE_MODE_YADIF:
  235. return obs_load_effect(&obs->video.deinterlace_yadif_effect,
  236. "deinterlace_yadif.effect");
  237. case OBS_DEINTERLACE_MODE_YADIF_2X:
  238. return obs_load_effect(&obs->video.deinterlace_yadif_2x_effect,
  239. "deinterlace_yadif_2x.effect");
  240. }
  241. return NULL;
  242. }
  243. #define TWOX_TOLERANCE 1000000
  244. void deinterlace_render(obs_source_t *s)
  245. {
  246. gs_effect_t *effect = s->deinterlace_effect;
  247. uint64_t frame2_ts;
  248. gs_eparam_t *image = gs_effect_get_param_by_name(effect, "image");
  249. gs_eparam_t *prev = gs_effect_get_param_by_name(effect,
  250. "previous_image");
  251. gs_eparam_t *field = gs_effect_get_param_by_name(effect, "field_order");
  252. gs_eparam_t *frame2 = gs_effect_get_param_by_name(effect, "frame2");
  253. gs_eparam_t *dimensions = gs_effect_get_param_by_name(effect,
  254. "dimensions");
  255. struct vec2 size = {(float)s->async_width, (float)s->async_height};
  256. gs_texture_t *cur_tex = s->async_texrender ?
  257. gs_texrender_get_texture(s->async_texrender) :
  258. s->async_texture;
  259. gs_texture_t *prev_tex = s->async_prev_texrender ?
  260. gs_texrender_get_texture(s->async_prev_texrender) :
  261. s->async_prev_texture;
  262. if (!cur_tex || !prev_tex || !s->async_width || !s->async_height)
  263. return;
  264. gs_effect_set_texture(image, cur_tex);
  265. gs_effect_set_texture(prev, prev_tex);
  266. gs_effect_set_int(field, s->deinterlace_top_first);
  267. gs_effect_set_vec2(dimensions, &size);
  268. frame2_ts = s->deinterlace_frame_ts + s->deinterlace_offset +
  269. s->deinterlace_half_duration - TWOX_TOLERANCE;
  270. gs_effect_set_bool(frame2, obs->video.video_time >= frame2_ts);
  271. while (gs_effect_loop(effect, "Draw"))
  272. gs_draw_sprite(NULL, s->async_flip ? GS_FLIP_V : 0,
  273. s->async_width, s->async_height);
  274. }
  275. static void enable_deinterlacing(obs_source_t *source,
  276. enum obs_deinterlace_mode mode)
  277. {
  278. obs_enter_graphics();
  279. if (source->async_format != VIDEO_FORMAT_NONE &&
  280. source->async_width != 0 &&
  281. source->async_height != 0)
  282. set_deinterlace_texture_size(source);
  283. source->deinterlace_mode = mode;
  284. source->deinterlace_effect = get_effect(mode);
  285. pthread_mutex_lock(&source->async_mutex);
  286. if (source->prev_async_frame) {
  287. remove_async_frame(source, source->prev_async_frame);
  288. source->prev_async_frame = NULL;
  289. }
  290. pthread_mutex_unlock(&source->async_mutex);
  291. obs_leave_graphics();
  292. }
  293. static void disable_deinterlacing(obs_source_t *source)
  294. {
  295. obs_enter_graphics();
  296. gs_texture_destroy(source->async_prev_texture);
  297. gs_texrender_destroy(source->async_prev_texrender);
  298. source->deinterlace_mode = OBS_DEINTERLACE_MODE_DISABLE;
  299. source->async_prev_texture = NULL;
  300. source->async_prev_texrender = NULL;
  301. obs_leave_graphics();
  302. }
  303. void obs_source_set_deinterlace_mode(obs_source_t *source,
  304. enum obs_deinterlace_mode mode)
  305. {
  306. if (!obs_source_valid(source, "obs_source_set_deinterlace_mode"))
  307. return;
  308. if (source->deinterlace_mode == mode)
  309. return;
  310. if (source->deinterlace_mode == OBS_DEINTERLACE_MODE_DISABLE) {
  311. enable_deinterlacing(source, mode);
  312. } else if (mode == OBS_DEINTERLACE_MODE_DISABLE) {
  313. disable_deinterlacing(source);
  314. } else {
  315. obs_enter_graphics();
  316. source->deinterlace_mode = mode;
  317. source->deinterlace_effect = get_effect(mode);
  318. obs_leave_graphics();
  319. }
  320. }
  321. enum obs_deinterlace_mode obs_source_get_deinterlace_mode(
  322. const obs_source_t *source)
  323. {
  324. return obs_source_valid(source, "obs_source_set_deinterlace_mode") ?
  325. source->deinterlace_mode : OBS_DEINTERLACE_MODE_DISABLE;
  326. }
  327. void obs_source_set_deinterlace_field_order(obs_source_t *source,
  328. enum obs_deinterlace_field_order field_order)
  329. {
  330. if (!obs_source_valid(source, "obs_source_set_deinterlace_field_order"))
  331. return;
  332. source->deinterlace_top_first =
  333. field_order == OBS_DEINTERLACE_FIELD_ORDER_TOP;
  334. }
  335. enum obs_deinterlace_field_order obs_source_get_deinterlace_field_order(
  336. const obs_source_t *source)
  337. {
  338. if (!obs_source_valid(source, "obs_source_set_deinterlace_field_order"))
  339. return OBS_DEINTERLACE_FIELD_ORDER_TOP;
  340. return source->deinterlace_top_first
  341. ? OBS_DEINTERLACE_FIELD_ORDER_TOP
  342. : OBS_DEINTERLACE_FIELD_ORDER_BOTTOM;
  343. }