函數原型如下:
int av_lockmgr_register (int(*cb)(void **mutex, enum AVLockOp op));其作用是注冊一個用戶自定義的lock manager, 其中cb可定義如下(ffplay.c):
static int lockmgr(void **mtx, enum AVLockOp op){ switch(op) { case AV_LOCK_CREATE: *mtx = SDL_CreateMutex(); if(!*mtx) { av_log(NULL, AV_LOG_FATAL, "SDL_CreateMutex(): %s/n", SDL_GetError()); return 1; } return 0; case AV_LOCK_OBTAIN: return !!SDL_LockMutex(*mtx); case AV_LOCK_RELEASE: return !!SDL_UnlockMutex(*mtx); case AV_LOCK_DESTROY: SDL_DestroyMutex(*mtx); return 0; } return 1;}通過傳入的op參數來創建、加鎖、解鎖以及銷毀操作。
當多個線程調用avcodec_open2、avcodec_close的時可能導致失敗。 從ffmpeg源碼可知,失敗的主要原因是在調用此函數時為確保函數為原子操作,在函數的開頭和結尾處使用了一個變量entangled_thread_counter來記錄當前函數是否已經有其他線程進入,如果有其他線程正在函數內運行,則會調用失敗。其解釋如下:
在avcodec_open2函數開頭有如下判斷, 當返回值小于0時返回。 ret = ff_lock_avcodec(avctx); if (ret < 0) return ret; ff_lock_avcodec定義如下,當lockmgr_cb不為空時調用該函數加鎖。當entangled_thread_counter大于1時,則返回AVERROR(EINVAL),說明有多個線程在該函數中運行。 int ff_lock_avcodec(AVCodecContext *log_ctx, const AVCodec *codec){ if (codec->caps_internal & FF_CODEC_CAP_INIT_THREADSAFE || !codec->init) return 0; if (lockmgr_cb) { if ((*lockmgr_cb)(&codec_mutex, AV_LOCK_OBTAIN)) return -1; } if (avPRiv_atomic_int_add_and_fetch(&entangled_thread_counter, 1) != 1) { av_log(log_ctx, AV_LOG_ERROR, "Insufficient thread locking. At least %d threads are " "calling avcodec_open2() at the same time right now./n", entangled_thread_counter); if (!lockmgr_cb) av_log(log_ctx, AV_LOG_ERROR, "No lock manager is set, please see av_lockmgr_register()/n"); ff_avcodec_locked = 1; ff_unlock_avcodec(codec); return AVERROR(EINVAL); } av_assert0(!ff_avcodec_locked); ff_avcodec_locked = 1; return 0;}新聞熱點
疑難解答