static void dump_source_info(struct ffmpeg_source *s, const char *input, const char *input_format, bool is_advanced) { FF_BLOG(LOG_INFO, "settings:\n" "\tinput: %s\n" "\tinput_format: %s\n" "\tis_looping: %s\n" "\tis_forcing_scale: %s\n" "\tis_hw_decoding: %s\n" "\tis_clear_on_media_end: %s\n" "\trestart_on_activate: %s", input ? input : "(null)", input_format ? input_format : "(null)", s->is_looping ? "yes" : "no", s->is_forcing_scale ? "yes" : "no", s->is_hw_decoding ? "yes" : "no", s->is_clear_on_media_end ? "yes" : "no", s->restart_on_activate ? "yes" : "no"); if (!is_advanced) return; FF_BLOG(LOG_INFO, "advanced settings:\n" "\taudio_buffer_size: %d\n" "\tvideo_buffer_size: %d\n" "\tframe_drop: %s", s->audio_buffer_size, s->video_buffer_size, frame_drop_to_str(s->frame_drop)); }
static bool set_obs_frame_colorprops(struct ff_frame *frame, struct ffmpeg_source *s, struct obs_source_frame *obs_frame) { enum AVColorSpace frame_cs = av_frame_get_colorspace(frame->frame); enum video_colorspace obs_cs; switch(frame_cs) { case AVCOL_SPC_BT709: obs_cs = VIDEO_CS_709; break; case AVCOL_SPC_SMPTE170M: case AVCOL_SPC_BT470BG: obs_cs = VIDEO_CS_601; break; case AVCOL_SPC_UNSPECIFIED: obs_cs = VIDEO_CS_DEFAULT; break; default: FF_BLOG(LOG_WARNING, "frame using an unsupported colorspace %d", frame_cs); obs_cs = VIDEO_CS_DEFAULT; } enum video_range_type range; obs_frame->format = ffmpeg_to_obs_video_format(frame->frame->format); obs_frame->full_range = frame->frame->color_range == AVCOL_RANGE_JPEG; if (s->range != VIDEO_RANGE_DEFAULT) obs_frame->full_range = s->range == VIDEO_RANGE_FULL; range = obs_frame->full_range ? VIDEO_RANGE_FULL : VIDEO_RANGE_PARTIAL; if (!video_format_get_parameters(obs_cs, range, obs_frame->color_matrix, obs_frame->color_range_min, obs_frame->color_range_max)) { FF_BLOG(LOG_ERROR, "Failed to get video format " "parameters for video format %u", obs_cs); return false; } return true; }
static void get_nb_frames(void *data, calldata_t *cd) { struct ffmpeg_source *s = data; int64_t frames = 0; if (!s->media.fmt) { calldata_set_int(cd, "num_frames", frames); return; } int video_stream_index = av_find_best_stream(s->media.fmt, AVMEDIA_TYPE_VIDEO, -1, -1, NULL, 0); if (video_stream_index < 0) { FF_BLOG(LOG_WARNING, "Getting number of frames failed: No " "video stream in media file!"); calldata_set_int(cd, "num_frames", frames); return; } AVStream *stream = s->media.fmt->streams[video_stream_index]; if (stream->nb_frames > 0) { frames = stream->nb_frames; } else { FF_BLOG(LOG_DEBUG, "nb_frames not set, estimating using frame " "rate and duration"); AVRational avg_frame_rate = stream->avg_frame_rate; frames = (int64_t)ceil((double)s->media.fmt->duration / (double)AV_TIME_BASE * (double)avg_frame_rate.num / (double)avg_frame_rate.den); } calldata_set_int(cd, "num_frames", frames); }
static void dump_source_info(struct ffmpeg_source *s, const char *input, const char *input_format) { FF_BLOG(LOG_INFO, "settings:\n" "\tinput: %s\n" "\tinput_format: %s\n" "\tis_looping: %s\n" "\tis_hw_decoding: %s\n" "\tis_clear_on_media_end: %s\n" "\trestart_on_activate: %s\n" "\tclose_when_inactive: %s", input ? input : "(null)", input_format ? input_format : "(null)", s->is_looping ? "yes" : "no", s->is_hw_decoding ? "yes" : "no", s->is_clear_on_media_end ? "yes" : "no", s->restart_on_activate ? "yes" : "no", s->close_when_inactive ? "yes" : "no"); }
bool update_sws_context(struct ffmpeg_source *s, AVFrame *frame) { if (frame->width != s->sws_width || frame->height != s->sws_height || frame->format != s->sws_format) { if (s->sws_ctx != NULL) sws_freeContext(s->sws_ctx); if (frame->width <= 0 || frame->height <= 0) { FF_BLOG(LOG_ERROR, "unable to create a sws " "context that has a width(%d) or " "height(%d) of zero.", frame->width, frame->height); goto fail; } s->sws_ctx = sws_getContext( frame->width, frame->height, frame->format, frame->width, frame->height, AV_PIX_FMT_BGRA, SWS_BILINEAR, NULL, NULL, NULL); if (s->sws_ctx == NULL) { FF_BLOG(LOG_ERROR, "unable to create sws " "context with src{w:%d,h:%d,f:%d}->" "dst{w:%d,h:%d,f:%d}", frame->width, frame->height, frame->format, frame->width, frame->height, AV_PIX_FMT_BGRA); goto fail; } if (s->sws_data != NULL) bfree(s->sws_data); s->sws_data = bzalloc(frame->width * frame->height * 4); if (s->sws_data == NULL) { FF_BLOG(LOG_ERROR, "unable to allocate sws " "pixel data with size %d", frame->width * frame->height * 4); goto fail; } s->sws_linesize = frame->width * 4; s->sws_width = frame->width; s->sws_height = frame->height; s->sws_format = frame->format; } return true; fail: if (s->sws_ctx != NULL) sws_freeContext(s->sws_ctx); s->sws_ctx = NULL; if (s->sws_data) bfree(s->sws_data); s->sws_data = NULL; s->sws_linesize = 0; s->sws_width = 0; s->sws_height = 0; s->sws_format = 0; return false; }
static void ffmpeg_source_update(void *data, obs_data_t *settings) { struct ffmpeg_source *s = data; bool is_local_file = obs_data_get_bool(settings, "is_local_file"); bool is_advanced = obs_data_get_bool(settings, "advanced"); char *input; char *input_format; bfree(s->input); bfree(s->input_format); if (is_local_file) { input = (char *)obs_data_get_string(settings, "local_file"); input_format = NULL; s->is_looping = obs_data_get_bool(settings, "looping"); } else { input = (char *)obs_data_get_string(settings, "input"); input_format = (char *)obs_data_get_string(settings, "input_format"); s->is_looping = false; } s->input = input ? bstrdup(input) : NULL; s->input_format = input_format ? bstrdup(input_format) : NULL; s->is_advanced = is_advanced; s->is_hw_decoding = obs_data_get_bool(settings, "hw_decode"); s->is_clear_on_media_end = obs_data_get_bool(settings, "clear_on_media_end"); s->restart_on_activate = obs_data_get_bool(settings, "restart_on_activate"); s->is_forcing_scale = true; if (is_advanced) { s->audio_buffer_size = (int)obs_data_get_int(settings, "audio_buffer_size"); s->video_buffer_size = (int)obs_data_get_int(settings, "video_buffer_size"); s->frame_drop = (enum AVDiscard)obs_data_get_int(settings, "frame_drop"); s->is_forcing_scale = obs_data_get_bool(settings, "force_scale"); if (s->audio_buffer_size < 1) { s->audio_buffer_size = 1; FF_BLOG(LOG_WARNING, "invalid audio_buffer_size %d", s->audio_buffer_size); } if (s->video_buffer_size < 1) { s->video_buffer_size = 1; FF_BLOG(LOG_WARNING, "invalid audio_buffer_size %d", s->audio_buffer_size); } if (s->frame_drop < AVDISCARD_NONE || s->frame_drop > AVDISCARD_ALL) { s->frame_drop = AVDISCARD_DEFAULT; FF_BLOG(LOG_WARNING, "invalid frame_drop %d", s->frame_drop); } } dump_source_info(s, input, input_format, is_advanced); if (!s->restart_on_activate || obs_source_active(s->source)) ffmpeg_source_start(s); }
static void ffmpeg_source_update(void *data, obs_data_t *settings) { struct ffmpeg_source *s = data; bool is_local_file = obs_data_get_bool(settings, "is_local_file"); bool is_advanced = obs_data_get_bool(settings, "advanced"); bool is_looping; char *input; char *input_format; if (is_local_file) { input = (char *)obs_data_get_string(settings, "local_file"); input_format = NULL; is_looping = obs_data_get_bool(settings, "looping"); } else { input = (char *)obs_data_get_string(settings, "input"); input_format = (char *)obs_data_get_string(settings, "input_format"); is_looping = false; } s->is_forcing_scale = obs_data_get_bool(settings, "force_scale"); s->is_hw_decoding = obs_data_get_bool(settings, "hw_decode"); s->is_clear_on_media_end = obs_data_get_bool(settings, "clear_on_media_end"); if (s->demuxer != NULL) ff_demuxer_free(s->demuxer); s->demuxer = ff_demuxer_init(); s->demuxer->options.is_hw_decoding = s->is_hw_decoding; s->demuxer->options.is_looping = is_looping; if (is_advanced) { int audio_buffer_size = (int)obs_data_get_int(settings, "audio_buffer_size"); int video_buffer_size = (int)obs_data_get_int(settings, "video_buffer_size"); enum AVDiscard frame_drop = (enum AVDiscard)obs_data_get_int(settings, "frame_drop"); if (audio_buffer_size < 1) { audio_buffer_size = 1; FF_BLOG(LOG_WARNING, "invalid audio_buffer_size %d", audio_buffer_size); } if (video_buffer_size < 1) { video_buffer_size = 1; FF_BLOG(LOG_WARNING, "invalid audio_buffer_size %d", audio_buffer_size); } s->demuxer->options.audio_frame_queue_size = audio_buffer_size; s->demuxer->options.video_frame_queue_size = video_buffer_size; if (frame_drop < AVDISCARD_NONE || frame_drop > AVDISCARD_ALL) { frame_drop = AVDISCARD_NONE; FF_BLOG(LOG_WARNING, "invalid frame_drop %d", frame_drop); } s->demuxer->options.frame_drop = frame_drop; } ff_demuxer_set_callbacks(&s->demuxer->video_callbacks, video_frame, NULL, NULL, NULL, NULL, s); ff_demuxer_set_callbacks(&s->demuxer->audio_callbacks, audio_frame, NULL, NULL, NULL, NULL, s); dump_source_info(s, input, input_format, is_advanced); ff_demuxer_open(s->demuxer, input, input_format); }