diff --git a/libobs/obs-source.c b/libobs/obs-source.c index 875197accf3fa0d46b433e2fd8bb0bfd96685213..eb1f61d423f12fb67d45b03515d22629a0f643a1 100644 --- a/libobs/obs-source.c +++ b/libobs/obs-source.c @@ -1501,9 +1501,14 @@ void obs_source_output_audio(obs_source_t source, static inline bool frame_out_of_bounds(obs_source_t source, uint64_t ts) { - return ((ts - source->last_frame_ts) > MAX_TIMESTAMP_JUMP); + if (ts < source->last_frame_ts) + return ((source->last_frame_ts - ts) > MAX_TIMESTAMP_JUMP); + else + return ((ts - source->last_frame_ts) > MAX_TIMESTAMP_JUMP); } +/* #define DEBUG_ASYNC_FRAMES 1 */ + static bool ready_async_frame(obs_source_t source, uint64_t sys_time) { struct obs_source_frame *next_frame = source->video_frames.array[0]; @@ -1512,26 +1517,60 @@ static bool ready_async_frame(obs_source_t source, uint64_t sys_time) uint64_t frame_time = next_frame->timestamp; uint64_t frame_offset = 0; +#if DEBUG_ASYNC_FRAMES + blog(LOG_DEBUG, "source->last_frame_ts: %llu, frame_time: %llu, " + "sys_offset: %llu, frame_offset: %llu, " + "number of frames: %lu", + source->last_frame_ts, frame_time, sys_offset, + frame_time - source->last_frame_ts, + (unsigned long)source->video_frames.num); +#endif + /* account for timestamp invalidation */ if (frame_out_of_bounds(source, frame_time)) { +#if DEBUG_ASYNC_FRAMES + blog(LOG_DEBUG, "timing jump"); +#endif source->last_frame_ts = next_frame->timestamp; + return true; } else { frame_offset = frame_time - source->last_frame_ts; - source->last_frame_ts += frame_offset; + source->last_frame_ts += sys_offset; } - while (frame_offset <= sys_offset) { + while (source->last_frame_ts > next_frame->timestamp) { + + /* this tries to reduce the needless frame duplication, also + * helps smooth out async rendering to frame boundaries. In + * other words, tries to keep the framerate as smooth as + * possible */ + if ((source->last_frame_ts - next_frame->timestamp) < 1000000) + break; + + if (frame) + da_erase(source->video_frames, 0); + +#if DEBUG_ASYNC_FRAMES + blog(LOG_DEBUG, "new frame, " + "source->last_frame_ts: %llu, " + "next_frame->timestamp: %llu", + source->last_frame_ts, + next_frame->timestamp); +#endif + obs_source_frame_destroy(frame); if (source->video_frames.num == 1) return true; frame = next_frame; - da_erase(source->video_frames, 0); - next_frame = source->video_frames.array[0]; + next_frame = source->video_frames.array[1]; /* more timestamp checking and compensating */ if ((next_frame->timestamp - frame_time) > MAX_TIMESTAMP_JUMP) { +#if DEBUG_ASYNC_FRAMES + blog(LOG_DEBUG, "timing jump"); +#endif source->last_frame_ts = next_frame->timestamp - frame_offset; } @@ -1540,7 +1579,10 @@ static bool ready_async_frame(obs_source_t source, uint64_t sys_time) frame_offset = frame_time - source->last_frame_ts; } - obs_source_frame_destroy(frame); +#if DEBUG_ASYNC_FRAMES + if (!frame) + blog(LOG_DEBUG, "no frame!"); +#endif return frame != NULL; } @@ -1573,11 +1615,11 @@ struct obs_source_frame *obs_source_get_frame(obs_source_t source) pthread_mutex_lock(&source->video_mutex); + sys_time = os_gettime_ns(); + if (!source->video_frames.num) goto unlock; - sys_time = os_gettime_ns(); - if (!source->last_frame_ts) { frame = source->video_frames.array[0]; da_erase(source->video_frames, 0); @@ -1589,13 +1631,16 @@ struct obs_source_frame *obs_source_get_frame(obs_source_t source) /* reset timing to current system time */ if (frame) { + uint64_t min_expected_sys_ts = + frame->timestamp + source->timing_adjust; + source->timing_adjust = sys_time - frame->timestamp; source->timing_set = true; } +unlock: source->last_sys_timestamp = sys_time; -unlock: pthread_mutex_unlock(&source->video_mutex); if (frame)