| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221 | #include "graphics.h"#include <libavcodec/avcodec.h>#include <libavformat/avformat.h>#include <libswscale/swscale.h>#include "../obs-ffmpeg-compat.h"struct ffmpeg_image {	const char         *file;	AVFormatContext    *fmt_ctx;	AVCodecContext     *decoder_ctx;	AVCodec            *decoder;	AVStream           *stream;	int                stream_idx;	int                cx, cy;	enum AVPixelFormat format;};static bool ffmpeg_image_open_decoder_context(struct ffmpeg_image *info){	int ret = av_find_best_stream(info->fmt_ctx, AVMEDIA_TYPE_VIDEO,			-1, 1, NULL, 0);	if (ret < 0) {		blog(LOG_WARNING, "Couldn't find video stream in file '%s': %s",				info->file, av_err2str(ret));		return false;	}	info->stream_idx  = ret;	info->stream      = info->fmt_ctx->streams[ret];	info->decoder_ctx = info->stream->codec;	info->decoder     = avcodec_find_decoder(info->decoder_ctx->codec_id);	if (!info->decoder) {		blog(LOG_WARNING, "Failed to find decoder for file '%s'",				info->file);		return false;	}	ret = avcodec_open2(info->decoder_ctx, info->decoder, NULL);	if (ret < 0) {		blog(LOG_WARNING, "Failed to open video codec for file '%s': "		                  "%s", info->file, av_err2str(ret));		return false;	}	return true;}static void ffmpeg_image_free(struct ffmpeg_image *info){	avcodec_close(info->decoder_ctx);	avformat_close_input(&info->fmt_ctx);}static bool ffmpeg_image_init(struct ffmpeg_image *info, const char *file){	int ret;	if (!file || !*file)		return false;	memset(info, 0, sizeof(struct ffmpeg_image));	info->file       = file;	info->stream_idx = -1;	ret = avformat_open_input(&info->fmt_ctx, file, NULL, NULL);	if (ret < 0) {		blog(LOG_WARNING, "Failed to open file '%s': %s",				info->file, av_err2str(ret));		return false;	}	ret = avformat_find_stream_info(info->fmt_ctx, NULL);	if (ret < 0) {		blog(LOG_WARNING, "Could not find stream info for file '%s':"		                  " %s", info->file, av_err2str(ret));		goto fail;	}	if (!ffmpeg_image_open_decoder_context(info))		goto fail;	info->cx     = info->decoder_ctx->width;	info->cy     = info->decoder_ctx->height;	info->format = info->decoder_ctx->pix_fmt;	return true;fail:	ffmpeg_image_free(info);	return false;}static bool ffmpeg_image_reformat_frame(struct ffmpeg_image *info,		AVFrame *frame, uint8_t *out, int linesize){	struct SwsContext *sws_ctx = NULL;	int               ret      = 0;	if (info->format == AV_PIX_FMT_RGBA ||	    info->format == AV_PIX_FMT_BGRA ||	    info->format == AV_PIX_FMT_BGR0) {		if (linesize != frame->linesize[0]) {			int min_line = linesize < frame->linesize[0] ?				linesize : frame->linesize[0];			for (int y = 0; y < info->cy; y++)				memcpy(out + y * linesize,				       frame->data[0] + y * frame->linesize[0],				       min_line);		} else {			memcpy(out, frame->data[0], linesize * info->cy);		}	} else {		sws_ctx = sws_getContext(info->cx, info->cy, info->format,				info->cx, info->cy, AV_PIX_FMT_BGRA,				SWS_POINT, NULL, NULL, NULL);		if (!sws_ctx) {			blog(LOG_WARNING, "Failed to create scale context "			                  "for '%s'", info->file);			return false;		}		ret = sws_scale(sws_ctx, (const uint8_t *const*)frame->data,				frame->linesize, 0, info->cy, &out, &linesize);		sws_freeContext(sws_ctx);		if (ret < 0) {			blog(LOG_WARNING, "sws_scale failed for '%s': %s",					info->file, av_err2str(ret));			return false;		}		info->format = AV_PIX_FMT_BGRA;	}	return true;}static bool ffmpeg_image_decode(struct ffmpeg_image *info, uint8_t *out,		int linesize){	AVPacket          packet    = {0};	bool              success   = false;	AVFrame           *frame    = av_frame_alloc();	int               got_frame = 0;	int               ret;	if (!frame) {		blog(LOG_WARNING, "Failed to create frame data for '%s'",				info->file);		return false;	}	ret = av_read_frame(info->fmt_ctx, &packet);	if (ret < 0) {		blog(LOG_WARNING, "Failed to read image frame from '%s': %s",				info->file, av_err2str(ret));		goto fail;	}	while (!got_frame) {		ret = avcodec_decode_video2(info->decoder_ctx, frame,				&got_frame, &packet);		if (ret < 0) {			blog(LOG_WARNING, "Failed to decode frame for '%s': %s",					info->file, av_err2str(ret));			goto fail;		}	}	success = ffmpeg_image_reformat_frame(info, frame, out, linesize);fail:	av_free_packet(&packet);	av_frame_free(&frame);	return success;}void gs_init_image_deps(void){	av_register_all();}void gs_free_image_deps(void){}static inline enum gs_color_format convert_format(enum AVPixelFormat format){	switch ((int)format) {	case AV_PIX_FMT_RGBA: return GS_RGBA;	case AV_PIX_FMT_BGRA: return GS_BGRA;	case AV_PIX_FMT_BGR0: return GS_BGRX;	}	return GS_BGRX;}texture_t gs_create_texture_from_file(const char *file){	struct ffmpeg_image image;	texture_t           tex = NULL;	if (ffmpeg_image_init(&image, file)) {		uint8_t *data = malloc(image.cx * image.cy * 4);		if (ffmpeg_image_decode(&image, data, image.cx * 4)) {			tex = gs_create_texture(image.cx, image.cy,					convert_format(image.format),					1, (const uint8_t**)&data, 0);		}		ffmpeg_image_free(&image);		free(data);	}	return tex;}
 |