FFmpeg coverage


Directory: ../../../ffmpeg/
File: src/libavcodec/pthread_frame.c
Date: 2026-04-23 02:20:26
Exec Total Coverage
Lines: 423 543 77.9%
Functions: 26 28 92.9%
Branches: 190 316 60.1%

Line Branch Exec Source
1 /*
2 * This file is part of FFmpeg.
3 *
4 * FFmpeg is free software; you can redistribute it and/or
5 * modify it under the terms of the GNU Lesser General Public
6 * License as published by the Free Software Foundation; either
7 * version 2.1 of the License, or (at your option) any later version.
8 *
9 * FFmpeg is distributed in the hope that it will be useful,
10 * but WITHOUT ANY WARRANTY; without even the implied warranty of
11 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
12 * Lesser General Public License for more details.
13 *
14 * You should have received a copy of the GNU Lesser General Public
15 * License along with FFmpeg; if not, write to the Free Software
16 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
17 */
18
19 /**
20 * @file
21 * Frame multithreading support functions
22 * @see doc/multithreading.txt
23 */
24
25 #include <stdatomic.h>
26
27 #include "avcodec.h"
28 #include "avcodec_internal.h"
29 #include "codec_desc.h"
30 #include "codec_internal.h"
31 #include "decode.h"
32 #include "hwaccel_internal.h"
33 #include "hwconfig.h"
34 #include "internal.h"
35 #include "packet_internal.h"
36 #include "pthread_internal.h"
37 #include "libavutil/refstruct.h"
38 #include "thread.h"
39 #include "threadframe.h"
40 #include "version_major.h"
41
42 #include "libavutil/avassert.h"
43 #include "libavutil/buffer.h"
44 #include "libavutil/common.h"
45 #include "libavutil/cpu.h"
46 #include "libavutil/frame.h"
47 #include "libavutil/internal.h"
48 #include "libavutil/log.h"
49 #include "libavutil/mem.h"
50 #include "libavutil/opt.h"
51 #include "libavutil/thread.h"
52
53 enum {
54 /// Set when the thread is awaiting a packet.
55 STATE_INPUT_READY,
56 /// Set before the codec has called ff_thread_finish_setup().
57 STATE_SETTING_UP,
58 /// Set after the codec has called ff_thread_finish_setup().
59 STATE_SETUP_FINISHED,
60 };
61
62 enum {
63 UNINITIALIZED, ///< Thread has not been created, AVCodec->close mustn't be called
64 NEEDS_CLOSE, ///< FFCodec->close needs to be called
65 INITIALIZED, ///< Thread has been properly set up
66 };
67
68 typedef struct DecodedFrames {
69 AVFrame **f;
70 size_t nb_f;
71 size_t nb_f_allocated;
72 } DecodedFrames;
73
74 typedef struct ThreadFrameProgress {
75 atomic_int progress[2];
76 } ThreadFrameProgress;
77
78 /**
79 * Context used by codec threads and stored in their AVCodecInternal thread_ctx.
80 */
81 typedef struct PerThreadContext {
82 struct FrameThreadContext *parent;
83
84 pthread_t thread;
85 int thread_init;
86 unsigned pthread_init_cnt;///< Number of successfully initialized mutexes/conditions
87 pthread_cond_t input_cond; ///< Used to wait for a new packet from the main thread.
88 pthread_cond_t progress_cond; ///< Used by child threads to wait for progress to change.
89 pthread_cond_t output_cond; ///< Used by the main thread to wait for frames to finish.
90
91 pthread_mutex_t mutex; ///< Mutex used to protect the contents of the PerThreadContext.
92 pthread_mutex_t progress_mutex; ///< Mutex used to protect frame progress values and progress_cond.
93
94 AVCodecContext *avctx; ///< Context used to decode packets passed to this thread.
95
96 AVPacket *avpkt; ///< Input packet (for decoding) or output (for encoding).
97
98 /**
99 * Decoded frames from a single decode iteration.
100 */
101 DecodedFrames df;
102 int result; ///< The result of the last codec decode/encode() call.
103
104 atomic_int state;
105
106 int die; ///< Set when the thread should exit.
107
108 int hwaccel_serializing;
109 int async_serializing;
110
111 // set to 1 in ff_thread_finish_setup() when a threadsafe hwaccel is used;
112 // cannot check hwaccel caps directly, because
113 // worked threads clear hwaccel state for thread-unsafe hwaccels
114 // after each decode call
115 int hwaccel_threadsafe;
116
117 atomic_int debug_threads; ///< Set if the FF_DEBUG_THREADS option is set.
118 } PerThreadContext;
119
120 /**
121 * Context stored in the client AVCodecInternal thread_ctx.
122 */
123 typedef struct FrameThreadContext {
124 PerThreadContext *threads; ///< The contexts for each thread.
125 PerThreadContext *prev_thread; ///< The last thread submit_packet() was called on.
126
127 unsigned pthread_init_cnt; ///< Number of successfully initialized mutexes/conditions
128 pthread_mutex_t buffer_mutex; ///< Mutex used to protect get/release_buffer().
129 /**
130 * This lock is used for ensuring threads run in serial when thread-unsafe
131 * hwaccel is used.
132 */
133 pthread_mutex_t hwaccel_mutex;
134 pthread_mutex_t async_mutex;
135 pthread_cond_t async_cond;
136 int async_lock;
137
138 DecodedFrames df;
139 int result;
140
141 /**
142 * Packet to be submitted to the next thread for decoding.
143 */
144 AVPacket *next_pkt;
145
146 int next_decoding; ///< The next context to submit a packet to.
147 int next_finished; ///< The next context to return output from.
148
149 /* hwaccel state for thread-unsafe hwaccels is temporarily stored here in
150 * order to transfer its ownership to the next decoding thread without the
151 * need for extra synchronization */
152 const AVHWAccel *stash_hwaccel;
153 void *stash_hwaccel_context;
154 void *stash_hwaccel_priv;
155 } FrameThreadContext;
156
157 1818 static int hwaccel_serial(const AVCodecContext *avctx)
158 {
159
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 1818 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
1818 return avctx->hwaccel && !(ffhwaccel(avctx->hwaccel)->caps_internal & HWACCEL_CAP_THREAD_SAFE);
160 }
161
162 1221 static void async_lock(FrameThreadContext *fctx)
163 {
164 1221 pthread_mutex_lock(&fctx->async_mutex);
165
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1221 times.
1221 while (fctx->async_lock)
166 pthread_cond_wait(&fctx->async_cond, &fctx->async_mutex);
167 1221 fctx->async_lock = 1;
168 1221 pthread_mutex_unlock(&fctx->async_mutex);
169 1221 }
170
171 1221 static void async_unlock(FrameThreadContext *fctx)
172 {
173 1221 pthread_mutex_lock(&fctx->async_mutex);
174
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1221 times.
1221 av_assert0(fctx->async_lock);
175 1221 fctx->async_lock = 0;
176 1221 pthread_cond_broadcast(&fctx->async_cond);
177 1221 pthread_mutex_unlock(&fctx->async_mutex);
178 1221 }
179
180 76 static void thread_set_name(PerThreadContext *p)
181 {
182 76 AVCodecContext *avctx = p->avctx;
183 76 int idx = p - p->parent->threads;
184 char name[16];
185
186 76 snprintf(name, sizeof(name), "av:%.7s:df%d", avctx->codec->name, idx);
187
188 76 ff_thread_setname(name);
189 76 }
190
191 // get a free frame to decode into
192 1144 static AVFrame *decoded_frames_get_free(DecodedFrames *df)
193 {
194
2/2
✓ Branch 0 taken 163 times.
✓ Branch 1 taken 981 times.
1144 if (df->nb_f == df->nb_f_allocated) {
195 163 AVFrame **tmp = av_realloc_array(df->f, df->nb_f + 1,
196 sizeof(*df->f));
197
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 163 times.
163 if (!tmp)
198 return NULL;
199 163 df->f = tmp;
200
201 163 df->f[df->nb_f] = av_frame_alloc();
202
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 163 times.
163 if (!df->f[df->nb_f])
203 return NULL;
204
205 163 df->nb_f_allocated++;
206 }
207
208
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1144 times.
1144 av_assert0(!df->f[df->nb_f]->buf[0]);
209
210 1144 return df->f[df->nb_f];
211 }
212
213 537 static void decoded_frames_pop(DecodedFrames *df, AVFrame *dst)
214 {
215 537 AVFrame *tmp_frame = df->f[0];
216 537 av_frame_move_ref(dst, tmp_frame);
217 537 memmove(df->f, df->f + 1, (df->nb_f - 1) * sizeof(*df->f));
218 537 df->f[--df->nb_f] = tmp_frame;
219 537 }
220
221 static void decoded_frames_flush(DecodedFrames *df)
222 {
223 for (size_t i = 0; i < df->nb_f; i++)
224 av_frame_unref(df->f[i]);
225 df->nb_f = 0;
226 }
227
228 86 static void decoded_frames_free(DecodedFrames *df)
229 {
230
2/2
✓ Branch 0 taken 163 times.
✓ Branch 1 taken 86 times.
249 for (size_t i = 0; i < df->nb_f_allocated; i++)
231 163 av_frame_free(&df->f[i]);
232 86 av_freep(&df->f);
233 86 df->nb_f = 0;
234 86 df->nb_f_allocated = 0;
235 86 }
236
237 /**
238 * Codec worker thread.
239 *
240 * Automatically calls ff_thread_finish_setup() if the codec does
241 * not provide an update_thread_context method, or if the codec returns
242 * before calling it.
243 */
244 76 static attribute_align_arg void *frame_worker_thread(void *arg)
245 {
246 76 PerThreadContext *p = arg;
247 76 AVCodecContext *avctx = p->avctx;
248 76 const FFCodec *codec = ffcodec(avctx->codec);
249
250 76 thread_set_name(p);
251
252 76 pthread_mutex_lock(&p->mutex);
253 606 while (1) {
254 int ret;
255
256
4/4
✓ Branch 0 taken 756 times.
✓ Branch 1 taken 606 times.
✓ Branch 2 taken 680 times.
✓ Branch 3 taken 76 times.
1362 while (atomic_load(&p->state) == STATE_INPUT_READY && !p->die)
257 680 pthread_cond_wait(&p->input_cond, &p->mutex);
258
259
2/2
✓ Branch 0 taken 76 times.
✓ Branch 1 taken 606 times.
682 if (p->die) break;
260
261
2/2
✓ Branch 0 taken 494 times.
✓ Branch 1 taken 112 times.
606 if (!codec->update_thread_context)
262 494 ff_thread_finish_setup(avctx);
263
264 /* If a decoder supports hwaccel, then it must call ff_get_format().
265 * Since that call must happen before ff_thread_finish_setup(), the
266 * decoder is required to implement update_thread_context() and call
267 * ff_thread_finish_setup() manually. Therefore the above
268 * ff_thread_finish_setup() call did not happen and hwaccel_serializing
269 * cannot be true here. */
270
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 av_assert0(!p->hwaccel_serializing);
271
272 /* if the previous thread uses thread-unsafe hwaccel then we take the
273 * lock to ensure the threads don't run concurrently */
274
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 606 times.
606 if (hwaccel_serial(avctx)) {
275 pthread_mutex_lock(&p->parent->hwaccel_mutex);
276 p->hwaccel_serializing = 1;
277 }
278
279 606 ret = 0;
280
2/2
✓ Branch 0 taken 1144 times.
✓ Branch 1 taken 606 times.
1750 while (ret >= 0) {
281 AVFrame *frame;
282
283 /* get the frame which will store the output */
284 1144 frame = decoded_frames_get_free(&p->df);
285
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1144 times.
1144 if (!frame) {
286 p->result = AVERROR(ENOMEM);
287 goto alloc_fail;
288 }
289
290 /* do the actual decoding */
291 1144 ret = ff_decode_receive_frame_internal(avctx, frame);
292
2/2
✓ Branch 0 taken 538 times.
✓ Branch 1 taken 606 times.
1144 if (ret == 0)
293 538 p->df.nb_f++;
294
2/4
✓ Branch 0 taken 606 times.
✗ Branch 1 not taken.
✗ Branch 2 not taken.
✓ Branch 3 taken 606 times.
606 else if (ret < 0 && frame->buf[0])
295 av_frame_unref(frame);
296
297
2/2
✓ Branch 0 taken 606 times.
✓ Branch 1 taken 538 times.
1144 p->result = (ret == AVERROR(EAGAIN)) ? 0 : ret;
298 }
299
300
2/2
✓ Branch 0 taken 578 times.
✓ Branch 1 taken 28 times.
606 if (atomic_load(&p->state) == STATE_SETTING_UP)
301 28 ff_thread_finish_setup(avctx);
302
303 578 alloc_fail:
304
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (p->hwaccel_serializing) {
305 /* wipe hwaccel state for thread-unsafe hwaccels to avoid stale
306 * pointers lying around;
307 * the state was transferred to FrameThreadContext in
308 * ff_thread_finish_setup(), so nothing is leaked */
309 avctx->hwaccel = NULL;
310 avctx->hwaccel_context = NULL;
311 avctx->internal->hwaccel_priv_data = NULL;
312
313 p->hwaccel_serializing = 0;
314 pthread_mutex_unlock(&p->parent->hwaccel_mutex);
315 }
316
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
606 av_assert0(!avctx->hwaccel ||
317 (ffhwaccel(avctx->hwaccel)->caps_internal & HWACCEL_CAP_THREAD_SAFE));
318
319
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (p->async_serializing) {
320 p->async_serializing = 0;
321
322 async_unlock(p->parent);
323 }
324
325 606 pthread_mutex_lock(&p->progress_mutex);
326
327 606 atomic_store(&p->state, STATE_INPUT_READY);
328
329 606 pthread_cond_broadcast(&p->progress_cond);
330 606 pthread_cond_signal(&p->output_cond);
331 606 pthread_mutex_unlock(&p->progress_mutex);
332 }
333 76 pthread_mutex_unlock(&p->mutex);
334
335 76 return NULL;
336 }
337
338 /**
339 * Update the next thread's AVCodecContext with values from the reference thread's context.
340 *
341 * @param dst The destination context.
342 * @param src The source context.
343 * @param for_user 0 if the destination is a codec thread, 1 if the destination is the user's thread
344 * @return 0 on success, negative error code on failure
345 */
346 1111 static int update_context_from_thread(AVCodecContext *dst, const AVCodecContext *src, int for_user)
347 {
348 1111 const FFCodec *const codec = ffcodec(dst->codec);
349 1111 int err = 0;
350
351
5/6
✓ Branch 0 taken 1111 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 555 times.
✓ Branch 3 taken 556 times.
✓ Branch 4 taken 108 times.
✓ Branch 5 taken 447 times.
1111 if (dst != src && (for_user || codec->update_thread_context)) {
352 664 dst->time_base = src->time_base;
353 664 dst->framerate = src->framerate;
354 664 dst->width = src->width;
355 664 dst->height = src->height;
356 664 dst->pix_fmt = src->pix_fmt;
357 664 dst->sw_pix_fmt = src->sw_pix_fmt;
358
359 664 dst->coded_width = src->coded_width;
360 664 dst->coded_height = src->coded_height;
361
362 664 dst->has_b_frames = src->has_b_frames;
363 664 dst->idct_algo = src->idct_algo;
364 #if FF_API_CODEC_PROPS
365 FF_DISABLE_DEPRECATION_WARNINGS
366 664 dst->properties = src->properties;
367 FF_ENABLE_DEPRECATION_WARNINGS
368 #endif
369
370 664 dst->bits_per_coded_sample = src->bits_per_coded_sample;
371 664 dst->sample_aspect_ratio = src->sample_aspect_ratio;
372
373 664 dst->profile = src->profile;
374 664 dst->level = src->level;
375
376 664 dst->bits_per_raw_sample = src->bits_per_raw_sample;
377 664 dst->color_primaries = src->color_primaries;
378
379 664 dst->alpha_mode = src->alpha_mode;
380
381 664 dst->color_trc = src->color_trc;
382 664 dst->colorspace = src->colorspace;
383 664 dst->color_range = src->color_range;
384 664 dst->chroma_sample_location = src->chroma_sample_location;
385
386 664 dst->sample_rate = src->sample_rate;
387 664 dst->sample_fmt = src->sample_fmt;
388 664 err = av_channel_layout_copy(&dst->ch_layout, &src->ch_layout);
389
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 664 times.
664 if (err < 0)
390 return err;
391
392
1/2
✓ Branch 0 taken 664 times.
✗ Branch 1 not taken.
664 if (!!dst->hw_frames_ctx != !!src->hw_frames_ctx ||
393
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 664 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
664 (dst->hw_frames_ctx && dst->hw_frames_ctx->data != src->hw_frames_ctx->data)) {
394 av_buffer_unref(&dst->hw_frames_ctx);
395
396 if (src->hw_frames_ctx) {
397 dst->hw_frames_ctx = av_buffer_ref(src->hw_frames_ctx);
398 if (!dst->hw_frames_ctx)
399 return AVERROR(ENOMEM);
400 }
401 }
402
403 664 dst->hwaccel_flags = src->hwaccel_flags;
404
405 664 av_refstruct_replace(&dst->internal->pool, src->internal->pool);
406 664 ff_decode_internal_sync(dst, src);
407 }
408
409
2/2
✓ Branch 0 taken 556 times.
✓ Branch 1 taken 555 times.
1111 if (for_user) {
410
2/2
✓ Branch 0 taken 58 times.
✓ Branch 1 taken 498 times.
556 if (codec->update_thread_context_for_user)
411 58 err = codec->update_thread_context_for_user(dst, src);
412 } else {
413 555 const PerThreadContext *p_src = src->internal->thread_ctx;
414 555 PerThreadContext *p_dst = dst->internal->thread_ctx;
415
416
2/2
✓ Branch 0 taken 108 times.
✓ Branch 1 taken 447 times.
555 if (codec->update_thread_context) {
417 108 err = codec->update_thread_context(dst, src);
418
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 108 times.
108 if (err < 0)
419 return err;
420 }
421
422 // reset dst hwaccel state if needed
423
3/6
✓ Branch 0 taken 555 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 555 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
✓ Branch 5 taken 555 times.
555 av_assert0(p_dst->hwaccel_threadsafe ||
424 (!dst->hwaccel && !dst->internal->hwaccel_priv_data));
425
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 555 times.
555 if (p_dst->hwaccel_threadsafe &&
426 (!p_src->hwaccel_threadsafe || dst->hwaccel != src->hwaccel)) {
427 ff_hwaccel_uninit(dst);
428 p_dst->hwaccel_threadsafe = 0;
429 }
430
431 // propagate hwaccel state for threadsafe hwaccels
432
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 555 times.
555 if (p_src->hwaccel_threadsafe) {
433 const FFHWAccel *hwaccel = ffhwaccel(src->hwaccel);
434 if (!dst->hwaccel) {
435 if (hwaccel->priv_data_size) {
436 av_assert0(hwaccel->update_thread_context);
437
438 dst->internal->hwaccel_priv_data =
439 av_mallocz(hwaccel->priv_data_size);
440 if (!dst->internal->hwaccel_priv_data)
441 return AVERROR(ENOMEM);
442 }
443 dst->hwaccel = src->hwaccel;
444 }
445 av_assert0(dst->hwaccel == src->hwaccel);
446
447 if (hwaccel->update_thread_context) {
448 err = hwaccel->update_thread_context(dst, src);
449 if (err < 0) {
450 av_log(dst, AV_LOG_ERROR, "Error propagating hwaccel state\n");
451 ff_hwaccel_uninit(dst);
452 return err;
453 }
454 }
455 p_dst->hwaccel_threadsafe = 1;
456 }
457 }
458
459 1111 return err;
460 }
461
462 /**
463 * Update the next thread's AVCodecContext with values set by the user.
464 *
465 * @param dst The destination context.
466 * @param src The source context.
467 * @return 0 on success, negative error code on failure
468 */
469 606 static int update_context_from_user(AVCodecContext *dst, const AVCodecContext *src)
470 {
471 int err;
472
473 606 dst->flags = src->flags;
474
475 606 dst->draw_horiz_band= src->draw_horiz_band;
476 606 dst->get_buffer2 = src->get_buffer2;
477
478 606 dst->opaque = src->opaque;
479 606 dst->debug = src->debug;
480
481 606 dst->slice_flags = src->slice_flags;
482 606 dst->flags2 = src->flags2;
483 606 dst->export_side_data = src->export_side_data;
484
485 606 dst->skip_loop_filter = src->skip_loop_filter;
486 606 dst->skip_idct = src->skip_idct;
487 606 dst->skip_frame = src->skip_frame;
488
489 606 dst->frame_num = src->frame_num;
490
491 606 av_packet_unref(dst->internal->last_pkt_props);
492 606 err = av_packet_copy_props(dst->internal->last_pkt_props, src->internal->last_pkt_props);
493
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (err < 0)
494 return err;
495
496 606 return 0;
497 }
498
499 606 static int submit_packet(PerThreadContext *p, AVCodecContext *user_avctx,
500 AVPacket *in_pkt)
501 {
502 606 FrameThreadContext *fctx = p->parent;
503 606 PerThreadContext *prev_thread = fctx->prev_thread;
504 606 const AVCodec *codec = p->avctx->codec;
505 int ret;
506
507 606 pthread_mutex_lock(&p->mutex);
508
509 606 av_packet_unref(p->avpkt);
510 606 av_packet_move_ref(p->avpkt, in_pkt);
511
512
3/4
✓ Branch 0 taken 68 times.
✓ Branch 1 taken 538 times.
✓ Branch 2 taken 68 times.
✗ Branch 3 not taken.
606 if (AVPACKET_IS_EMPTY(p->avpkt))
513 68 p->avctx->internal->draining = 1;
514
515 606 ret = update_context_from_user(p->avctx, user_avctx);
516
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (ret) {
517 pthread_mutex_unlock(&p->mutex);
518 return ret;
519 }
520 606 atomic_store_explicit(&p->debug_threads,
521 (p->avctx->debug & FF_DEBUG_THREADS) != 0,
522 memory_order_relaxed);
523
524
2/2
✓ Branch 0 taken 596 times.
✓ Branch 1 taken 10 times.
606 if (prev_thread) {
525
2/2
✓ Branch 0 taken 347 times.
✓ Branch 1 taken 249 times.
596 if (atomic_load(&prev_thread->state) == STATE_SETTING_UP) {
526 347 pthread_mutex_lock(&prev_thread->progress_mutex);
527
2/2
✓ Branch 0 taken 347 times.
✓ Branch 1 taken 347 times.
694 while (atomic_load(&prev_thread->state) == STATE_SETTING_UP)
528 347 pthread_cond_wait(&prev_thread->progress_cond, &prev_thread->progress_mutex);
529 347 pthread_mutex_unlock(&prev_thread->progress_mutex);
530 }
531
532 /* codecs without delay might not be prepared to be called repeatedly here during
533 * flushing (vp3/theora), and also don't need to be, since from this point on, they
534 * will always return EOF anyway */
535
2/2
✓ Branch 0 taken 68 times.
✓ Branch 1 taken 528 times.
596 if (!p->avctx->internal->draining ||
536
2/2
✓ Branch 0 taken 27 times.
✓ Branch 1 taken 41 times.
68 (codec->capabilities & AV_CODEC_CAP_DELAY)) {
537 555 ret = update_context_from_thread(p->avctx, prev_thread->avctx, 0);
538
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 555 times.
555 if (ret) {
539 pthread_mutex_unlock(&p->mutex);
540 return ret;
541 }
542 }
543 }
544
545 /* transfer the stashed hwaccel state, if any */
546
1/4
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
✗ Branch 2 not taken.
✗ Branch 3 not taken.
606 av_assert0(!p->avctx->hwaccel || p->hwaccel_threadsafe);
547
1/2
✓ Branch 0 taken 606 times.
✗ Branch 1 not taken.
606 if (!p->hwaccel_threadsafe) {
548 606 FFSWAP(const AVHWAccel*, p->avctx->hwaccel, fctx->stash_hwaccel);
549 606 FFSWAP(void*, p->avctx->hwaccel_context, fctx->stash_hwaccel_context);
550 606 FFSWAP(void*, p->avctx->internal->hwaccel_priv_data, fctx->stash_hwaccel_priv);
551 }
552
553 606 atomic_store(&p->state, STATE_SETTING_UP);
554 606 pthread_cond_signal(&p->input_cond);
555 606 pthread_mutex_unlock(&p->mutex);
556
557 606 fctx->prev_thread = p;
558 606 fctx->next_decoding = (fctx->next_decoding + 1) % p->avctx->thread_count;
559
560 606 return 0;
561 }
562
563 1211 int ff_thread_receive_frame(AVCodecContext *avctx, AVFrame *frame, unsigned flags)
564 {
565 1211 FrameThreadContext *fctx = avctx->internal->thread_ctx;
566 1211 int ret = 0;
567
568 /* release the async lock, permitting blocked hwaccel threads to
569 * go forward while we are in this function */
570 1211 async_unlock(fctx);
571
572 /* submit packets to threads while there are no buffered results to return */
573
4/4
✓ Branch 0 taken 1282 times.
✓ Branch 1 taken 537 times.
✓ Branch 2 taken 1273 times.
✓ Branch 3 taken 9 times.
1819 while (!fctx->df.nb_f && !fctx->result) {
574 PerThreadContext *p;
575
576
2/2
✓ Branch 0 taken 1251 times.
✓ Branch 1 taken 22 times.
1273 if (fctx->next_decoding != fctx->next_finished &&
577
2/2
✓ Branch 0 taken 2 times.
✓ Branch 1 taken 1249 times.
1251 (flags & AV_CODEC_RECEIVE_FRAME_FLAG_SYNCHRONOUS))
578 2 goto wait_for_result;
579
580 /* get a packet to be submitted to the next thread */
581 1271 av_packet_unref(fctx->next_pkt);
582 1271 ret = ff_decode_get_packet(avctx, fctx->next_pkt);
583
4/4
✓ Branch 0 taken 733 times.
✓ Branch 1 taken 538 times.
✓ Branch 2 taken 665 times.
✓ Branch 3 taken 68 times.
1271 if (ret < 0 && ret != AVERROR_EOF)
584 665 goto finish;
585
586 606 ret = submit_packet(&fctx->threads[fctx->next_decoding], avctx,
587 fctx->next_pkt);
588
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (ret < 0)
589 goto finish;
590
591 /* do not return any frames until all threads have something to do */
592
2/2
✓ Branch 0 taken 74 times.
✓ Branch 1 taken 532 times.
606 if (fctx->next_decoding != fctx->next_finished &&
593
2/2
✓ Branch 0 taken 12 times.
✓ Branch 1 taken 62 times.
74 !avctx->internal->draining)
594 62 continue;
595
596 544 wait_for_result:
597 546 p = &fctx->threads[fctx->next_finished];
598 546 fctx->next_finished = (fctx->next_finished + 1) % avctx->thread_count;
599
600
2/2
✓ Branch 0 taken 149 times.
✓ Branch 1 taken 397 times.
546 if (atomic_load(&p->state) != STATE_INPUT_READY) {
601 149 pthread_mutex_lock(&p->progress_mutex);
602
2/2
✓ Branch 0 taken 149 times.
✓ Branch 1 taken 149 times.
298 while (atomic_load_explicit(&p->state, memory_order_relaxed) != STATE_INPUT_READY)
603 149 pthread_cond_wait(&p->output_cond, &p->progress_mutex);
604 149 pthread_mutex_unlock(&p->progress_mutex);
605 }
606
607 546 update_context_from_thread(avctx, p->avctx, 1);
608 546 fctx->result = p->result;
609 546 p->result = 0;
610
2/2
✓ Branch 0 taken 535 times.
✓ Branch 1 taken 11 times.
546 if (p->df.nb_f)
611 535 FFSWAP(DecodedFrames, fctx->df, p->df);
612 }
613
614 /* a thread may return multiple frames AND an error
615 * we first return all the frames, then the error */
616
2/2
✓ Branch 0 taken 537 times.
✓ Branch 1 taken 9 times.
546 if (fctx->df.nb_f) {
617 537 decoded_frames_pop(&fctx->df, frame);
618 537 ret = 0;
619 } else {
620 9 ret = fctx->result;
621 9 fctx->result = 0;
622 }
623
624 1211 finish:
625 1211 async_lock(fctx);
626 1211 return ret;
627 }
628
629 295222 void ff_thread_report_progress(ThreadFrame *f, int n, int field)
630 {
631 PerThreadContext *p;
632
2/2
✓ Branch 0 taken 1240 times.
✓ Branch 1 taken 293982 times.
295222 atomic_int *progress = f->progress ? f->progress->progress : NULL;
633
634
2/2
✓ Branch 0 taken 1240 times.
✓ Branch 1 taken 293982 times.
295222 if (!progress ||
635
2/2
✓ Branch 0 taken 27 times.
✓ Branch 1 taken 1213 times.
1240 atomic_load_explicit(&progress[field], memory_order_relaxed) >= n)
636 294009 return;
637
638 1213 p = f->owner[field]->internal->thread_ctx;
639
640
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 1213 times.
1213 if (atomic_load_explicit(&p->debug_threads, memory_order_relaxed))
641 av_log(f->owner[field], AV_LOG_DEBUG,
642 "%p finished %d field %d\n", progress, n, field);
643
644 1213 pthread_mutex_lock(&p->progress_mutex);
645
646 1213 atomic_store_explicit(&progress[field], n, memory_order_release);
647
648 1213 pthread_cond_broadcast(&p->progress_cond);
649 1213 pthread_mutex_unlock(&p->progress_mutex);
650 }
651
652 150876 void ff_thread_await_progress(const ThreadFrame *f, int n, int field)
653 {
654 PerThreadContext *p;
655
2/2
✓ Branch 0 taken 150193 times.
✓ Branch 1 taken 683 times.
150876 atomic_int *progress = f->progress ? f->progress->progress : NULL;
656
657
2/2
✓ Branch 0 taken 150193 times.
✓ Branch 1 taken 683 times.
150876 if (!progress ||
658
2/2
✓ Branch 0 taken 149649 times.
✓ Branch 1 taken 544 times.
150193 atomic_load_explicit(&progress[field], memory_order_acquire) >= n)
659 150332 return;
660
661 544 p = f->owner[field]->internal->thread_ctx;
662
663
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 544 times.
544 if (atomic_load_explicit(&p->debug_threads, memory_order_relaxed))
664 av_log(f->owner[field], AV_LOG_DEBUG,
665 "thread awaiting %d field %d from %p\n", n, field, progress);
666
667 544 pthread_mutex_lock(&p->progress_mutex);
668
2/2
✓ Branch 0 taken 546 times.
✓ Branch 1 taken 544 times.
1090 while (atomic_load_explicit(&progress[field], memory_order_relaxed) < n)
669 546 pthread_cond_wait(&p->progress_cond, &p->progress_mutex);
670 544 pthread_mutex_unlock(&p->progress_mutex);
671 }
672
673 24089 void ff_thread_finish_setup(AVCodecContext *avctx) {
674 PerThreadContext *p;
675
676
2/2
✓ Branch 0 taken 23483 times.
✓ Branch 1 taken 606 times.
24089 if (!(avctx->active_thread_type&FF_THREAD_FRAME)) return;
677
678 606 p = avctx->internal->thread_ctx;
679
680
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 p->hwaccel_threadsafe = avctx->hwaccel &&
681 (ffhwaccel(avctx->hwaccel)->caps_internal & HWACCEL_CAP_THREAD_SAFE);
682
683
1/4
✗ Branch 1 not taken.
✓ Branch 2 taken 606 times.
✗ Branch 3 not taken.
✗ Branch 4 not taken.
606 if (hwaccel_serial(avctx) && !p->hwaccel_serializing) {
684 pthread_mutex_lock(&p->parent->hwaccel_mutex);
685 p->hwaccel_serializing = 1;
686 }
687
688 /* this assumes that no hwaccel calls happen before ff_thread_finish_setup() */
689
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if (avctx->hwaccel &&
690 !(ffhwaccel(avctx->hwaccel)->caps_internal & HWACCEL_CAP_ASYNC_SAFE)) {
691 p->async_serializing = 1;
692
693 async_lock(p->parent);
694 }
695
696 /* thread-unsafe hwaccels share a single private data instance, so we
697 * save hwaccel state for passing to the next thread;
698 * this is done here so that this worker thread can wipe its own hwaccel
699 * state after decoding, without requiring synchronization */
700
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 av_assert0(!p->parent->stash_hwaccel);
701
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 606 times.
606 if (hwaccel_serial(avctx)) {
702 p->parent->stash_hwaccel = avctx->hwaccel;
703 p->parent->stash_hwaccel_context = avctx->hwaccel_context;
704 p->parent->stash_hwaccel_priv = avctx->internal->hwaccel_priv_data;
705 }
706
707 606 pthread_mutex_lock(&p->progress_mutex);
708
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 606 times.
606 if(atomic_load(&p->state) == STATE_SETUP_FINISHED){
709 av_log(avctx, AV_LOG_WARNING, "Multiple ff_thread_finish_setup() calls\n");
710 }
711
712 606 atomic_store(&p->state, STATE_SETUP_FINISHED);
713
714 606 pthread_cond_broadcast(&p->progress_cond);
715 606 pthread_mutex_unlock(&p->progress_mutex);
716 }
717
718 /// Waits for all threads to finish.
719 10 static av_cold void park_frame_worker_threads(FrameThreadContext *fctx, int thread_count)
720 {
721 int i;
722
723 10 async_unlock(fctx);
724
725
2/2
✓ Branch 0 taken 76 times.
✓ Branch 1 taken 10 times.
86 for (i = 0; i < thread_count; i++) {
726 76 PerThreadContext *p = &fctx->threads[i];
727
728
2/2
✓ Branch 0 taken 7 times.
✓ Branch 1 taken 69 times.
76 if (atomic_load(&p->state) != STATE_INPUT_READY) {
729 7 pthread_mutex_lock(&p->progress_mutex);
730
2/2
✓ Branch 0 taken 7 times.
✓ Branch 1 taken 7 times.
14 while (atomic_load(&p->state) != STATE_INPUT_READY)
731 7 pthread_cond_wait(&p->output_cond, &p->progress_mutex);
732 7 pthread_mutex_unlock(&p->progress_mutex);
733 }
734 }
735
736 10 async_lock(fctx);
737 10 }
738
739 #define OFF(member) offsetof(FrameThreadContext, member)
740 DEFINE_OFFSET_ARRAY(FrameThreadContext, thread_ctx, pthread_init_cnt,
741 (OFF(buffer_mutex), OFF(hwaccel_mutex), OFF(async_mutex)),
742 (OFF(async_cond)));
743 #undef OFF
744
745 #define OFF(member) offsetof(PerThreadContext, member)
746 DEFINE_OFFSET_ARRAY(PerThreadContext, per_thread, pthread_init_cnt,
747 (OFF(progress_mutex), OFF(mutex)),
748 (OFF(input_cond), OFF(progress_cond), OFF(output_cond)));
749 #undef OFF
750
751 10 av_cold void ff_frame_thread_free(AVCodecContext *avctx, int thread_count)
752 {
753 10 FrameThreadContext *fctx = avctx->internal->thread_ctx;
754 10 const FFCodec *codec = ffcodec(avctx->codec);
755 int i;
756
757 10 park_frame_worker_threads(fctx, thread_count);
758
759
2/2
✓ Branch 0 taken 76 times.
✓ Branch 1 taken 10 times.
86 for (i = 0; i < thread_count; i++) {
760 76 PerThreadContext *p = &fctx->threads[i];
761 76 AVCodecContext *ctx = p->avctx;
762
763
1/2
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
76 if (ctx->internal) {
764
1/2
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
76 if (p->thread_init == INITIALIZED) {
765 76 pthread_mutex_lock(&p->mutex);
766 76 p->die = 1;
767 76 pthread_cond_signal(&p->input_cond);
768 76 pthread_mutex_unlock(&p->mutex);
769
770 76 pthread_join(p->thread, NULL);
771 }
772
2/4
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
✓ Branch 2 taken 76 times.
✗ Branch 3 not taken.
76 if (codec->close && p->thread_init != UNINITIALIZED)
773 76 codec->close(ctx);
774
775 /* When using a threadsafe hwaccel, this is where
776 * each thread's context is uninit'd and freed. */
777 76 ff_hwaccel_uninit(ctx);
778
779
1/2
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
76 if (ctx->priv_data) {
780
2/2
✓ Branch 0 taken 58 times.
✓ Branch 1 taken 18 times.
76 if (codec->p.priv_class)
781 58 av_opt_free(ctx->priv_data);
782 76 av_freep(&ctx->priv_data);
783 }
784
785 76 av_refstruct_unref(&ctx->internal->pool);
786 76 av_packet_free(&ctx->internal->in_pkt);
787 76 av_packet_free(&ctx->internal->last_pkt_props);
788 76 ff_decode_internal_uninit(ctx);
789 76 av_freep(&ctx->internal);
790 76 av_buffer_unref(&ctx->hw_frames_ctx);
791 76 av_frame_side_data_free(&ctx->decoded_side_data,
792 &ctx->nb_decoded_side_data);
793 }
794
795 76 decoded_frames_free(&p->df);
796
797 76 ff_pthread_free(p, per_thread_offsets);
798 76 av_packet_free(&p->avpkt);
799
800 76 av_freep(&p->avctx);
801 }
802
803 10 decoded_frames_free(&fctx->df);
804 10 av_packet_free(&fctx->next_pkt);
805
806 10 av_freep(&fctx->threads);
807 10 ff_pthread_free(fctx, thread_ctx_offsets);
808
809 /* if we have stashed hwaccel state, move it to the user-facing context,
810 * so it will be freed in ff_codec_close() */
811
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 av_assert0(!avctx->hwaccel);
812 10 FFSWAP(const AVHWAccel*, avctx->hwaccel, fctx->stash_hwaccel);
813 10 FFSWAP(void*, avctx->hwaccel_context, fctx->stash_hwaccel_context);
814 10 FFSWAP(void*, avctx->internal->hwaccel_priv_data, fctx->stash_hwaccel_priv);
815
816 10 av_freep(&avctx->internal->thread_ctx);
817 10 }
818
819 76 static av_cold int init_thread(PerThreadContext *p, int *threads_to_free,
820 FrameThreadContext *fctx, AVCodecContext *avctx,
821 const FFCodec *codec, int first)
822 {
823 AVCodecContext *copy;
824 int err;
825
826 76 atomic_init(&p->state, STATE_INPUT_READY);
827
828 76 copy = av_memdup(avctx, sizeof(*avctx));
829
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (!copy)
830 return AVERROR(ENOMEM);
831 76 copy->priv_data = NULL;
832 76 copy->decoded_side_data = NULL;
833 76 copy->nb_decoded_side_data = 0;
834
835 /* From now on, this PerThreadContext will be cleaned up by
836 * ff_frame_thread_free in case of errors. */
837 76 (*threads_to_free)++;
838
839 76 p->parent = fctx;
840 76 p->avctx = copy;
841
842 76 copy->internal = ff_decode_internal_alloc();
843
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (!copy->internal)
844 return AVERROR(ENOMEM);
845 76 ff_decode_internal_sync(copy, avctx);
846 76 copy->internal->thread_ctx = p;
847 76 copy->internal->progress_frame_pool = avctx->internal->progress_frame_pool;
848
849 76 copy->delay = avctx->delay;
850
851
1/2
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
76 if (codec->priv_data_size) {
852 76 copy->priv_data = av_mallocz(codec->priv_data_size);
853
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (!copy->priv_data)
854 return AVERROR(ENOMEM);
855
856
2/2
✓ Branch 0 taken 58 times.
✓ Branch 1 taken 18 times.
76 if (codec->p.priv_class) {
857 58 *(const AVClass **)copy->priv_data = codec->p.priv_class;
858 58 err = av_opt_copy(copy->priv_data, avctx->priv_data);
859
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 58 times.
58 if (err < 0)
860 return err;
861 }
862 }
863
864 76 err = ff_pthread_init(p, per_thread_offsets);
865
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (err < 0)
866 return err;
867
868
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 76 times.
76 if (!(p->avpkt = av_packet_alloc()))
869 return AVERROR(ENOMEM);
870
871 76 copy->internal->is_frame_mt = 1;
872
2/2
✓ Branch 0 taken 66 times.
✓ Branch 1 taken 10 times.
76 if (!first)
873 66 copy->internal->is_copy = 1;
874
875 76 copy->internal->in_pkt = av_packet_alloc();
876
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (!copy->internal->in_pkt)
877 return AVERROR(ENOMEM);
878
879 76 copy->internal->last_pkt_props = av_packet_alloc();
880
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (!copy->internal->last_pkt_props)
881 return AVERROR(ENOMEM);
882
883
1/2
✓ Branch 0 taken 76 times.
✗ Branch 1 not taken.
76 if (codec->init) {
884 76 err = codec->init(copy);
885
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (err < 0) {
886 if (codec->caps_internal & FF_CODEC_CAP_INIT_CLEANUP)
887 p->thread_init = NEEDS_CLOSE;
888 return err;
889 }
890 }
891 76 p->thread_init = NEEDS_CLOSE;
892
893
2/2
✓ Branch 0 taken 10 times.
✓ Branch 1 taken 66 times.
76 if (first) {
894 10 update_context_from_thread(avctx, copy, 1);
895
896 10 av_frame_side_data_free(&avctx->decoded_side_data, &avctx->nb_decoded_side_data);
897
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 for (int i = 0; i < copy->nb_decoded_side_data; i++) {
898 err = av_frame_side_data_clone(&avctx->decoded_side_data,
899 &avctx->nb_decoded_side_data,
900 copy->decoded_side_data[i], 0);
901 if (err < 0)
902 return err;
903 }
904 }
905
906 76 atomic_init(&p->debug_threads, (copy->debug & FF_DEBUG_THREADS) != 0);
907
908 76 err = AVERROR(pthread_create(&p->thread, NULL, frame_worker_thread, p));
909
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (err < 0)
910 return err;
911 76 p->thread_init = INITIALIZED;
912
913 76 return 0;
914 }
915
916 10 av_cold int ff_frame_thread_init(AVCodecContext *avctx)
917 {
918 10 int thread_count = avctx->thread_count;
919 10 const FFCodec *codec = ffcodec(avctx->codec);
920 FrameThreadContext *fctx;
921 10 int err, i = 0;
922
923
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (!thread_count) {
924 int nb_cpus = av_cpu_count();
925 // use number of cores + 1 as thread count if there is more than one
926 if (nb_cpus > 1)
927 thread_count = avctx->thread_count = FFMIN(nb_cpus + 1, MAX_AUTO_THREADS);
928 else
929 thread_count = avctx->thread_count = 1;
930 }
931
932
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (thread_count <= 1) {
933 avctx->active_thread_type = 0;
934 return 0;
935 }
936
937 10 avctx->internal->thread_ctx = fctx = av_mallocz(sizeof(FrameThreadContext));
938
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (!fctx)
939 return AVERROR(ENOMEM);
940
941 10 err = ff_pthread_init(fctx, thread_ctx_offsets);
942
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (err < 0) {
943 ff_pthread_free(fctx, thread_ctx_offsets);
944 av_freep(&avctx->internal->thread_ctx);
945 return err;
946 }
947
948 10 fctx->next_pkt = av_packet_alloc();
949
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (!fctx->next_pkt)
950 return AVERROR(ENOMEM);
951
952 10 fctx->async_lock = 1;
953
954
2/2
✓ Branch 0 taken 6 times.
✓ Branch 1 taken 4 times.
10 if (codec->p.type == AVMEDIA_TYPE_VIDEO)
955 6 avctx->delay = avctx->thread_count - 1;
956
957 10 fctx->threads = av_calloc(thread_count, sizeof(*fctx->threads));
958
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 10 times.
10 if (!fctx->threads) {
959 err = AVERROR(ENOMEM);
960 goto error;
961 }
962
963
2/2
✓ Branch 0 taken 76 times.
✓ Branch 1 taken 10 times.
86 for (; i < thread_count; ) {
964 76 PerThreadContext *p = &fctx->threads[i];
965 76 int first = !i;
966
967 76 err = init_thread(p, &i, fctx, avctx, codec, first);
968
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 76 times.
76 if (err < 0)
969 goto error;
970 }
971
972 10 return 0;
973
974 error:
975 ff_frame_thread_free(avctx, i);
976 return err;
977 }
978
979 av_cold void ff_thread_flush(AVCodecContext *avctx)
980 {
981 int i;
982 FrameThreadContext *fctx = avctx->internal->thread_ctx;
983
984 if (!fctx) return;
985
986 park_frame_worker_threads(fctx, avctx->thread_count);
987 if (fctx->prev_thread) {
988 if (fctx->prev_thread != &fctx->threads[0])
989 update_context_from_thread(fctx->threads[0].avctx, fctx->prev_thread->avctx, 0);
990 }
991
992 fctx->next_decoding = fctx->next_finished = 0;
993 fctx->prev_thread = NULL;
994
995 decoded_frames_flush(&fctx->df);
996 fctx->result = 0;
997
998 for (i = 0; i < avctx->thread_count; i++) {
999 PerThreadContext *p = &fctx->threads[i];
1000
1001 decoded_frames_flush(&p->df);
1002 p->result = 0;
1003
1004 avcodec_flush_buffers(p->avctx);
1005 }
1006 }
1007
1008 37779 int ff_thread_can_start_frame(AVCodecContext *avctx)
1009 {
1010
2/2
✓ Branch 0 taken 52 times.
✓ Branch 1 taken 37727 times.
37779 if ((avctx->active_thread_type & FF_THREAD_FRAME) &&
1011
1/2
✓ Branch 1 taken 52 times.
✗ Branch 2 not taken.
52 ffcodec(avctx->codec)->update_thread_context) {
1012 52 PerThreadContext *p = avctx->internal->thread_ctx;
1013
1014
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 52 times.
52 if (atomic_load(&p->state) != STATE_SETTING_UP)
1015 return 0;
1016 }
1017
1018 37779 return 1;
1019 }
1020
1021 53159 static int thread_get_buffer_internal(AVCodecContext *avctx, AVFrame *f, int flags)
1022 {
1023 PerThreadContext *p;
1024 int err;
1025
1026
2/2
✓ Branch 0 taken 52621 times.
✓ Branch 1 taken 538 times.
53159 if (!(avctx->active_thread_type & FF_THREAD_FRAME))
1027 52621 return ff_get_buffer(avctx, f, flags);
1028
1029 538 p = avctx->internal->thread_ctx;
1030
2/2
✓ Branch 0 taken 453 times.
✓ Branch 1 taken 85 times.
538 if (atomic_load(&p->state) != STATE_SETTING_UP &&
1031
1/2
✗ Branch 1 not taken.
✓ Branch 2 taken 453 times.
453 ffcodec(avctx->codec)->update_thread_context) {
1032 av_log(avctx, AV_LOG_ERROR, "get_buffer() cannot be called after ff_thread_finish_setup()\n");
1033 return -1;
1034 }
1035
1036 538 pthread_mutex_lock(&p->parent->buffer_mutex);
1037 538 err = ff_get_buffer(avctx, f, flags);
1038
1039 538 pthread_mutex_unlock(&p->parent->buffer_mutex);
1040
1041 538 return err;
1042 }
1043
1044 53159 int ff_thread_get_buffer(AVCodecContext *avctx, AVFrame *f, int flags)
1045 {
1046 53159 int ret = thread_get_buffer_internal(avctx, f, flags);
1047
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 53159 times.
53159 if (ret < 0)
1048 av_log(avctx, AV_LOG_ERROR, "thread_get_buffer() failed\n");
1049 53159 return ret;
1050 }
1051
1052 26334 int ff_thread_get_ext_buffer(AVCodecContext *avctx, ThreadFrame *f, int flags)
1053 {
1054 int ret;
1055
1056 26334 f->owner[0] = f->owner[1] = avctx;
1057
2/2
✓ Branch 0 taken 26292 times.
✓ Branch 1 taken 42 times.
26334 if (!(avctx->active_thread_type & FF_THREAD_FRAME))
1058 26292 return ff_get_buffer(avctx, f->f, flags);
1059
1060 42 f->progress = av_refstruct_allocz(sizeof(*f->progress));
1061
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 42 times.
42 if (!f->progress)
1062 return AVERROR(ENOMEM);
1063
1064 42 atomic_init(&f->progress->progress[0], -1);
1065 42 atomic_init(&f->progress->progress[1], -1);
1066
1067 42 ret = ff_thread_get_buffer(avctx, f->f, flags);
1068
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 42 times.
42 if (ret)
1069 av_refstruct_unref(&f->progress);
1070 42 return ret;
1071 }
1072
1073 54629 void ff_thread_release_ext_buffer(ThreadFrame *f)
1074 {
1075 54629 av_refstruct_unref(&f->progress);
1076 54629 f->owner[0] = f->owner[1] = NULL;
1077
1/2
✓ Branch 0 taken 54629 times.
✗ Branch 1 not taken.
54629 if (f->f)
1078 54629 av_frame_unref(f->f);
1079 54629 }
1080
1081 749 av_cold enum ThreadingStatus ff_thread_sync_ref(AVCodecContext *avctx, size_t offset)
1082 {
1083 PerThreadContext *p;
1084 const void *ref;
1085
1086
2/2
✓ Branch 0 taken 741 times.
✓ Branch 1 taken 8 times.
749 if (!avctx->internal->is_copy)
1087 741 return avctx->active_thread_type & FF_THREAD_FRAME ?
1088
2/2
✓ Branch 0 taken 1 times.
✓ Branch 1 taken 740 times.
741 FF_THREAD_IS_FIRST_THREAD : FF_THREAD_NO_FRAME_THREADING;
1089
1090 8 p = avctx->internal->thread_ctx;
1091
1092 av_assert1(memcpy(&ref, (char*)avctx->priv_data + offset, sizeof(ref)) && ref == NULL);
1093
1094 8 memcpy(&ref, (const char*)p->parent->threads[0].avctx->priv_data + offset, sizeof(ref));
1095 av_assert1(ref);
1096 8 av_refstruct_replace((char*)avctx->priv_data + offset, ref);
1097
1098 8 return FF_THREAD_IS_COPY;
1099 }
1100
1101 1076 int ff_thread_get_packet(AVCodecContext *avctx, AVPacket *pkt)
1102 {
1103 1076 PerThreadContext *p = avctx->internal->thread_ctx;
1104
1105
3/4
✓ Branch 0 taken 538 times.
✓ Branch 1 taken 538 times.
✗ Branch 2 not taken.
✓ Branch 3 taken 538 times.
1076 if (!AVPACKET_IS_EMPTY(p->avpkt)) {
1106 538 av_packet_move_ref(pkt, p->avpkt);
1107 538 return 0;
1108 }
1109
1110
1/2
✗ Branch 0 not taken.
✓ Branch 1 taken 538 times.
538 return avctx->internal->draining ? AVERROR_EOF : AVERROR(EAGAIN);
1111 }
1112