diff --git a/libobs/obs-internal.h b/libobs/obs-internal.h
index 3fe9a9d0091267269ed555126af9e46abae62bd1..080afa8d32ea9f33dc1fc2f557491640efefb98a 100644
--- a/libobs/obs-internal.h
+++ b/libobs/obs-internal.h
@@ -293,6 +293,7 @@ struct obs_source {
 	uint64_t                        next_audio_ts_min;
 	uint64_t                        last_frame_ts;
 	uint64_t                        last_sys_timestamp;
+	bool                            async_rendered;
 
 	/* audio */
 	bool                            audio_failed;
diff --git a/libobs/obs-source.c b/libobs/obs-source.c
index 9816d5e9b04f89fa0b90c46b6200d154f5060442..875197accf3fa0d46b433e2fd8bb0bfd96685213 100644
--- a/libobs/obs-source.c
+++ b/libobs/obs-source.c
@@ -503,6 +503,8 @@ void obs_source_video_tick(obs_source_t source, float seconds)
 
 	if (source->context.data && source->info.video_tick)
 		source->info.video_tick(source->context.data, seconds);
+
+	source->async_rendered = false;
 }
 
 /* unless the value is 3+ hours worth of frames, this won't overflow */
@@ -1031,18 +1033,22 @@ static void obs_source_draw_async_texture(struct obs_source *source)
 
 static void obs_source_render_async_video(obs_source_t source)
 {
-	struct obs_source_frame *frame = obs_source_get_frame(source);
-	if (frame) {
-		if (!set_async_texture_size(source, frame))
-			return;
-		if (!update_async_texture(source, frame))
-			return;
+	if (!source->async_rendered) {
+		struct obs_source_frame *frame = obs_source_get_frame(source);
+
+		source->async_rendered = true;
+		if (frame) {
+			if (!set_async_texture_size(source, frame))
+				return;
+			if (!update_async_texture(source, frame))
+				return;
+		}
+
+		obs_source_release_frame(source, frame);
 	}
 
 	if (source->async_texture)
 		obs_source_draw_async_texture(source);
-
-	obs_source_release_frame(source, frame);
 }
 
 static inline void obs_source_render_filters(obs_source_t source)