Welcome to mirror list, hosted at ThFree Co, Russian Federation.

git.blender.org/blender.git - Unnamed repository; edit this file 'description' to name the repository.
summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorCampbell Barton <ideasman42@gmail.com>2019-04-17 07:17:24 +0300
committerCampbell Barton <ideasman42@gmail.com>2019-04-17 07:21:24 +0300
commite12c08e8d170b7ca40f204a5b0423c23a9fbc2c1 (patch)
tree8cf3453d12edb177a218ef8009357518ec6cab6a /source/blender/imbuf/intern/anim_movie.c
parentb3dabc200a4b0399ec6b81f2ff2730d07b44fcaa (diff)
ClangFormat: apply to source, most of intern
Apply clang format as proposed in T53211. For details on usage and instructions for migrating branches without conflicts, see: https://wiki.blender.org/wiki/Tools/ClangFormat
Diffstat (limited to 'source/blender/imbuf/intern/anim_movie.c')
-rw-r--r--source/blender/imbuf/intern/anim_movie.c2342
1 files changed, 1165 insertions, 1177 deletions
diff --git a/source/blender/imbuf/intern/anim_movie.c b/source/blender/imbuf/intern/anim_movie.c
index 43d91dc0d8e..13813fb23b3 100644
--- a/source/blender/imbuf/intern/anim_movie.c
+++ b/source/blender/imbuf/intern/anim_movie.c
@@ -21,24 +21,28 @@
* \ingroup imbuf
*/
-
#ifdef _WIN32
-#define INC_OLE2
-#include <windows.h>
-#include <windowsx.h>
-#include <mmsystem.h>
-#include <memory.h>
-#include <commdlg.h>
-#include <vfw.h>
-
-#undef AVIIF_KEYFRAME /* redefined in AVI_avi.h */
-#undef AVIIF_LIST /* redefined in AVI_avi.h */
-
-#define FIXCC(fcc) \
- { \
- if (fcc == 0) { fcc = mmioFOURCC('N', 'o', 'n', 'e'); } \
- if (fcc == BI_RLE8) { fcc = mmioFOURCC('R', 'l', 'e', '8'); } \
- } (void)0
+# define INC_OLE2
+# include <windows.h>
+# include <windowsx.h>
+# include <mmsystem.h>
+# include <memory.h>
+# include <commdlg.h>
+# include <vfw.h>
+
+# undef AVIIF_KEYFRAME /* redefined in AVI_avi.h */
+# undef AVIIF_LIST /* redefined in AVI_avi.h */
+
+# define FIXCC(fcc) \
+ { \
+ if (fcc == 0) { \
+ fcc = mmioFOURCC('N', 'o', 'n', 'e'); \
+ } \
+ if (fcc == BI_RLE8) { \
+ fcc = mmioFOURCC('R', 'l', 'e', '8'); \
+ } \
+ } \
+ (void)0
#endif
@@ -49,9 +53,9 @@
#include <math.h>
#include <limits.h>
#ifndef _WIN32
-#include <dirent.h>
+# include <dirent.h>
#else
-#include <io.h>
+# include <io.h>
#endif
#include "BLI_utildefines.h"
@@ -75,7 +79,7 @@
#include "IMB_metadata.h"
#ifdef WITH_FFMPEG
-# include "BKE_global.h" /* ENDIAN_ORDER */
+# include "BKE_global.h" /* ENDIAN_ORDER */
# include <libavformat/avformat.h>
# include <libavcodec/avcodec.h>
@@ -83,28 +87,27 @@
# include <libswscale/swscale.h>
# include "ffmpeg_compat.h"
-#endif //WITH_FFMPEG
+#endif //WITH_FFMPEG
int ismovie(const char *UNUSED(filepath))
{
- return 0;
+ return 0;
}
/* never called, just keep the linker happy */
static int startmovie(struct anim *UNUSED(anim))
{
- return 1;
+ return 1;
}
static ImBuf *movie_fetchibuf(struct anim *UNUSED(anim), int UNUSED(position))
{
- return NULL;
+ return NULL;
}
static void free_anim_movie(struct anim *UNUSED(anim))
{
- /* pass */
+ /* pass */
}
-
#if defined(_WIN32)
# define PATHSEPARATOR '\\'
#else
@@ -113,84 +116,88 @@ static void free_anim_movie(struct anim *UNUSED(anim))
static int an_stringdec(const char *string, char *head, char *tail, unsigned short *numlen)
{
- unsigned short len, nume, nums = 0;
- short i;
- bool found = false;
-
- len = strlen(string);
- nume = len;
-
- for (i = len - 1; i >= 0; i--) {
- if (string[i] == PATHSEPARATOR) break;
- if (isdigit(string[i])) {
- if (found) {
- nums = i;
- }
- else {
- nume = i;
- nums = i;
- found = true;
- }
- }
- else {
- if (found) break;
- }
- }
- if (found) {
- strcpy(tail, &string[nume + 1]);
- strcpy(head, string);
- head[nums] = '\0';
- *numlen = nume - nums + 1;
- return ((int)atoi(&(string[nums])));
- }
- tail[0] = '\0';
- strcpy(head, string);
- *numlen = 0;
- return true;
+ unsigned short len, nume, nums = 0;
+ short i;
+ bool found = false;
+
+ len = strlen(string);
+ nume = len;
+
+ for (i = len - 1; i >= 0; i--) {
+ if (string[i] == PATHSEPARATOR)
+ break;
+ if (isdigit(string[i])) {
+ if (found) {
+ nums = i;
+ }
+ else {
+ nume = i;
+ nums = i;
+ found = true;
+ }
+ }
+ else {
+ if (found)
+ break;
+ }
+ }
+ if (found) {
+ strcpy(tail, &string[nume + 1]);
+ strcpy(head, string);
+ head[nums] = '\0';
+ *numlen = nume - nums + 1;
+ return ((int)atoi(&(string[nums])));
+ }
+ tail[0] = '\0';
+ strcpy(head, string);
+ *numlen = 0;
+ return true;
}
-
-static void an_stringenc(char *string, const char *head, const char *tail, unsigned short numlen, int pic)
+static void an_stringenc(
+ char *string, const char *head, const char *tail, unsigned short numlen, int pic)
{
- BLI_stringenc(string, head, tail, numlen, pic);
+ BLI_stringenc(string, head, tail, numlen, pic);
}
#ifdef WITH_AVI
static void free_anim_avi(struct anim *anim)
{
-#if defined(_WIN32)
- int i;
-#endif
-
- if (anim == NULL) return;
- if (anim->avi == NULL) return;
-
- AVI_close(anim->avi);
- MEM_freeN(anim->avi);
- anim->avi = NULL;
-
-#if defined(_WIN32)
-
- if (anim->pgf) {
- AVIStreamGetFrameClose(anim->pgf);
- anim->pgf = NULL;
- }
-
- for (i = 0; i < anim->avistreams; i++) {
- AVIStreamRelease(anim->pavi[i]);
- }
- anim->avistreams = 0;
-
- if (anim->pfileopen) {
- AVIFileRelease(anim->pfile);
- anim->pfileopen = 0;
- AVIFileExit();
- }
-#endif
-
- anim->duration = 0;
+# if defined(_WIN32)
+ int i;
+# endif
+
+ if (anim == NULL)
+ return;
+ if (anim->avi == NULL)
+ return;
+
+ AVI_close(anim->avi);
+ MEM_freeN(anim->avi);
+ anim->avi = NULL;
+
+# if defined(_WIN32)
+
+ if (anim->pgf) {
+ AVIStreamGetFrameClose(anim->pgf);
+ anim->pgf = NULL;
+ }
+
+ for (i = 0; i < anim->avistreams; i++) {
+ AVIStreamRelease(anim->pavi[i]);
+ }
+ anim->avistreams = 0;
+
+ if (anim->pfileopen) {
+ AVIFileRelease(anim->pfile);
+ anim->pfileopen = 0;
+ AVIFileExit();
+ }
+# endif
+
+ anim->duration = 0;
}
-#endif /* WITH_AVI */
+#endif /* WITH_AVI */
#ifdef WITH_FFMPEG
static void free_anim_ffmpeg(struct anim *anim);
@@ -198,487 +205,496 @@ static void free_anim_ffmpeg(struct anim *anim);
void IMB_free_anim(struct anim *anim)
{
- if (anim == NULL) {
- printf("free anim, anim == NULL\n");
- return;
- }
+ if (anim == NULL) {
+ printf("free anim, anim == NULL\n");
+ return;
+ }
- free_anim_movie(anim);
+ free_anim_movie(anim);
#ifdef WITH_AVI
- free_anim_avi(anim);
+ free_anim_avi(anim);
#endif
#ifdef WITH_FFMPEG
- free_anim_ffmpeg(anim);
+ free_anim_ffmpeg(anim);
#endif
- IMB_free_indices(anim);
- IMB_metadata_free(anim->metadata);
+ IMB_free_indices(anim);
+ IMB_metadata_free(anim->metadata);
- MEM_freeN(anim);
+ MEM_freeN(anim);
}
void IMB_close_anim(struct anim *anim)
{
- if (anim == NULL) return;
+ if (anim == NULL)
+ return;
- IMB_free_anim(anim);
+ IMB_free_anim(anim);
}
void IMB_close_anim_proxies(struct anim *anim)
{
- if (anim == NULL)
- return;
+ if (anim == NULL)
+ return;
- IMB_free_indices(anim);
+ IMB_free_indices(anim);
}
struct IDProperty *IMB_anim_load_metadata(struct anim *anim)
{
- switch (anim->curtype) {
- case ANIM_FFMPEG:
- {
+ switch (anim->curtype) {
+ case ANIM_FFMPEG: {
#ifdef WITH_FFMPEG
- AVDictionaryEntry *entry = NULL;
+ AVDictionaryEntry *entry = NULL;
- BLI_assert(anim->pFormatCtx != NULL);
- av_log(anim->pFormatCtx, AV_LOG_DEBUG, "METADATA FETCH\n");
+ BLI_assert(anim->pFormatCtx != NULL);
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "METADATA FETCH\n");
- while (true) {
- entry = av_dict_get(anim->pFormatCtx->metadata, "", entry, AV_DICT_IGNORE_SUFFIX);
- if (entry == NULL) break;
+ while (true) {
+ entry = av_dict_get(anim->pFormatCtx->metadata, "", entry, AV_DICT_IGNORE_SUFFIX);
+ if (entry == NULL)
+ break;
- /* Delay creation of the property group until there is actual metadata to put in there. */
- IMB_metadata_ensure(&anim->metadata);
- IMB_metadata_set_field(anim->metadata, entry->key, entry->value);
- }
+ /* Delay creation of the property group until there is actual metadata to put in there. */
+ IMB_metadata_ensure(&anim->metadata);
+ IMB_metadata_set_field(anim->metadata, entry->key, entry->value);
+ }
#endif
- break;
- }
- case ANIM_SEQUENCE:
- case ANIM_AVI:
- case ANIM_MOVIE:
- /* TODO */
- break;
- case ANIM_NONE:
- default:
- break;
- }
- return anim->metadata;
+ break;
+ }
+ case ANIM_SEQUENCE:
+ case ANIM_AVI:
+ case ANIM_MOVIE:
+ /* TODO */
+ break;
+ case ANIM_NONE:
+ default:
+ break;
+ }
+ return anim->metadata;
}
-struct anim *IMB_open_anim(const char *name, int ib_flags, int streamindex, char colorspace[IM_MAX_SPACE])
+struct anim *IMB_open_anim(const char *name,
+ int ib_flags,
+ int streamindex,
+ char colorspace[IM_MAX_SPACE])
{
- struct anim *anim;
-
- BLI_assert(!BLI_path_is_rel(name));
-
- anim = (struct anim *)MEM_callocN(sizeof(struct anim), "anim struct");
- if (anim != NULL) {
- if (colorspace) {
- colorspace_set_default_role(colorspace, IM_MAX_SPACE, COLOR_ROLE_DEFAULT_BYTE);
- BLI_strncpy(anim->colorspace, colorspace, sizeof(anim->colorspace));
- }
- else {
- colorspace_set_default_role(anim->colorspace, sizeof(anim->colorspace), COLOR_ROLE_DEFAULT_BYTE);
- }
-
- BLI_strncpy(anim->name, name, sizeof(anim->name));
- anim->ib_flags = ib_flags;
- anim->streamindex = streamindex;
- }
- return(anim);
+ struct anim *anim;
+
+ BLI_assert(!BLI_path_is_rel(name));
+
+ anim = (struct anim *)MEM_callocN(sizeof(struct anim), "anim struct");
+ if (anim != NULL) {
+ if (colorspace) {
+ colorspace_set_default_role(colorspace, IM_MAX_SPACE, COLOR_ROLE_DEFAULT_BYTE);
+ BLI_strncpy(anim->colorspace, colorspace, sizeof(anim->colorspace));
+ }
+ else {
+ colorspace_set_default_role(
+ anim->colorspace, sizeof(anim->colorspace), COLOR_ROLE_DEFAULT_BYTE);
+ }
+
+ BLI_strncpy(anim->name, name, sizeof(anim->name));
+ anim->ib_flags = ib_flags;
+ anim->streamindex = streamindex;
+ }
+ return (anim);
}
void IMB_suffix_anim(struct anim *anim, const char *suffix)
{
- BLI_strncpy(anim->suffix, suffix, sizeof(anim->suffix));
+ BLI_strncpy(anim->suffix, suffix, sizeof(anim->suffix));
}
#ifdef WITH_AVI
static int startavi(struct anim *anim)
{
- AviError avierror;
-#if defined(_WIN32)
- HRESULT hr;
- int i, firstvideo = -1;
- int streamcount;
- BYTE abFormat[1024];
- LONG l;
- LPBITMAPINFOHEADER lpbi;
- AVISTREAMINFO avis;
-
- streamcount = anim->streamindex;
-#endif
-
- anim->avi = MEM_callocN(sizeof(AviMovie), "animavi");
-
- if (anim->avi == NULL) {
- printf("Can't open avi: %s\n", anim->name);
- return -1;
- }
-
- avierror = AVI_open_movie(anim->name, anim->avi);
-
-#if defined(_WIN32)
- if (avierror == AVI_ERROR_COMPRESSION) {
- AVIFileInit();
- hr = AVIFileOpen(&anim->pfile, anim->name, OF_READ, 0L);
- if (hr == 0) {
- anim->pfileopen = 1;
- for (i = 0; i < MAXNUMSTREAMS; i++) {
- if (AVIFileGetStream(anim->pfile, &anim->pavi[i], 0L, i) != AVIERR_OK) {
- break;
- }
-
- AVIStreamInfo(anim->pavi[i], &avis, sizeof(avis));
- if ((avis.fccType == streamtypeVIDEO) && (firstvideo == -1)) {
- if (streamcount > 0) {
- streamcount--;
- continue;
- }
- anim->pgf = AVIStreamGetFrameOpen(anim->pavi[i], NULL);
- if (anim->pgf) {
- firstvideo = i;
-
- /* get stream length */
- anim->avi->header->TotalFrames = AVIStreamLength(anim->pavi[i]);
-
- /* get information about images inside the stream */
- l = sizeof(abFormat);
- AVIStreamReadFormat(anim->pavi[i], 0, &abFormat, &l);
- lpbi = (LPBITMAPINFOHEADER)abFormat;
- anim->avi->header->Height = lpbi->biHeight;
- anim->avi->header->Width = lpbi->biWidth;
- }
- else {
- FIXCC(avis.fccHandler);
- FIXCC(avis.fccType);
- printf("Can't find AVI decoder for type : %4.4hs/%4.4hs\n",
- (LPSTR)&avis.fccType,
- (LPSTR)&avis.fccHandler);
- }
- }
- }
-
- /* register number of opened avistreams */
- anim->avistreams = i;
-
- /*
- * Couldn't get any video streams out of this file
- */
- if ((anim->avistreams == 0) || (firstvideo == -1)) {
- avierror = AVI_ERROR_FORMAT;
- }
- else {
- avierror = AVI_ERROR_NONE;
- anim->firstvideo = firstvideo;
- }
- }
- else {
- AVIFileExit();
- }
- }
-#endif
-
- if (avierror != AVI_ERROR_NONE) {
- AVI_print_error(avierror);
- printf("Error loading avi: %s\n", anim->name);
- free_anim_avi(anim);
- return -1;
- }
-
- anim->duration = anim->avi->header->TotalFrames;
- anim->params = NULL;
-
- anim->x = anim->avi->header->Width;
- anim->y = anim->avi->header->Height;
- anim->interlacing = 0;
- anim->orientation = 0;
- anim->framesize = anim->x * anim->y * 4;
-
- anim->curposition = 0;
- anim->preseek = 0;
-
- /* printf("x:%d y:%d size:%d interl:%d dur:%d\n", anim->x, anim->y, anim->framesize, anim->interlacing, anim->duration);*/
-
- return 0;
+ AviError avierror;
+# if defined(_WIN32)
+ HRESULT hr;
+ int i, firstvideo = -1;
+ int streamcount;
+ BYTE abFormat[1024];
+ LONG l;
+ LPBITMAPINFOHEADER lpbi;
+ AVISTREAMINFO avis;
+
+ streamcount = anim->streamindex;
+# endif
+
+ anim->avi = MEM_callocN(sizeof(AviMovie), "animavi");
+
+ if (anim->avi == NULL) {
+ printf("Can't open avi: %s\n", anim->name);
+ return -1;
+ }
+
+ avierror = AVI_open_movie(anim->name, anim->avi);
+
+# if defined(_WIN32)
+ if (avierror == AVI_ERROR_COMPRESSION) {
+ AVIFileInit();
+ hr = AVIFileOpen(&anim->pfile, anim->name, OF_READ, 0L);
+ if (hr == 0) {
+ anim->pfileopen = 1;
+ for (i = 0; i < MAXNUMSTREAMS; i++) {
+ if (AVIFileGetStream(anim->pfile, &anim->pavi[i], 0L, i) != AVIERR_OK) {
+ break;
+ }
+
+ AVIStreamInfo(anim->pavi[i], &avis, sizeof(avis));
+ if ((avis.fccType == streamtypeVIDEO) && (firstvideo == -1)) {
+ if (streamcount > 0) {
+ streamcount--;
+ continue;
+ }
+ anim->pgf = AVIStreamGetFrameOpen(anim->pavi[i], NULL);
+ if (anim->pgf) {
+ firstvideo = i;
+
+ /* get stream length */
+ anim->avi->header->TotalFrames = AVIStreamLength(anim->pavi[i]);
+
+ /* get information about images inside the stream */
+ l = sizeof(abFormat);
+ AVIStreamReadFormat(anim->pavi[i], 0, &abFormat, &l);
+ lpbi = (LPBITMAPINFOHEADER)abFormat;
+ anim->avi->header->Height = lpbi->biHeight;
+ anim->avi->header->Width = lpbi->biWidth;
+ }
+ else {
+ FIXCC(avis.fccHandler);
+ FIXCC(avis.fccType);
+ printf("Can't find AVI decoder for type : %4.4hs/%4.4hs\n",
+ (LPSTR)&avis.fccType,
+ (LPSTR)&avis.fccHandler);
+ }
+ }
+ }
+
+ /* register number of opened avistreams */
+ anim->avistreams = i;
+
+ /*
+ * Couldn't get any video streams out of this file
+ */
+ if ((anim->avistreams == 0) || (firstvideo == -1)) {
+ avierror = AVI_ERROR_FORMAT;
+ }
+ else {
+ avierror = AVI_ERROR_NONE;
+ anim->firstvideo = firstvideo;
+ }
+ }
+ else {
+ AVIFileExit();
+ }
+ }
+# endif
+
+ if (avierror != AVI_ERROR_NONE) {
+ AVI_print_error(avierror);
+ printf("Error loading avi: %s\n", anim->name);
+ free_anim_avi(anim);
+ return -1;
+ }
+
+ anim->duration = anim->avi->header->TotalFrames;
+ anim->params = NULL;
+
+ anim->x = anim->avi->header->Width;
+ anim->y = anim->avi->header->Height;
+ anim->interlacing = 0;
+ anim->orientation = 0;
+ anim->framesize = anim->x * anim->y * 4;
+
+ anim->curposition = 0;
+ anim->preseek = 0;
+
+ /* printf("x:%d y:%d size:%d interl:%d dur:%d\n", anim->x, anim->y, anim->framesize, anim->interlacing, anim->duration);*/
+
+ return 0;
}
-#endif /* WITH_AVI */
+#endif /* WITH_AVI */
#ifdef WITH_AVI
static ImBuf *avi_fetchibuf(struct anim *anim, int position)
{
- ImBuf *ibuf = NULL;
- int *tmp;
- int y;
-
- if (anim == NULL) {
- return NULL;
- }
-
-#if defined(_WIN32)
- if (anim->avistreams) {
- LPBITMAPINFOHEADER lpbi;
-
- if (anim->pgf) {
- lpbi = AVIStreamGetFrame(anim->pgf, position + AVIStreamStart(anim->pavi[anim->firstvideo]));
- if (lpbi) {
- ibuf = IMB_ibImageFromMemory((const unsigned char *) lpbi, 100, IB_rect, anim->colorspace, "<avi_fetchibuf>");
-//Oh brother...
- }
- }
- }
- else
-#endif
- {
- ibuf = IMB_allocImBuf(anim->x, anim->y, 24, IB_rect);
-
- tmp = AVI_read_frame(anim->avi, AVI_FORMAT_RGB32, position,
- AVI_get_stream(anim->avi, AVIST_VIDEO, 0));
-
- if (tmp == NULL) {
- printf("Error reading frame from AVI: '%s'\n", anim->name);
- IMB_freeImBuf(ibuf);
- return NULL;
- }
-
- for (y = 0; y < anim->y; y++) {
- memcpy(&(ibuf->rect)[((anim->y - y) - 1) * anim->x], &tmp[y * anim->x],
- anim->x * 4);
- }
-
- MEM_freeN(tmp);
- }
-
- ibuf->rect_colorspace = colormanage_colorspace_get_named(anim->colorspace);
-
- return ibuf;
+ ImBuf *ibuf = NULL;
+ int *tmp;
+ int y;
+
+ if (anim == NULL) {
+ return NULL;
+ }
+
+# if defined(_WIN32)
+ if (anim->avistreams) {
+ LPBITMAPINFOHEADER lpbi;
+
+ if (anim->pgf) {
+ lpbi = AVIStreamGetFrame(anim->pgf, position + AVIStreamStart(anim->pavi[anim->firstvideo]));
+ if (lpbi) {
+ ibuf = IMB_ibImageFromMemory(
+ (const unsigned char *)lpbi, 100, IB_rect, anim->colorspace, "<avi_fetchibuf>");
+ //Oh brother...
+ }
+ }
+ }
+ else
+# endif
+ {
+ ibuf = IMB_allocImBuf(anim->x, anim->y, 24, IB_rect);
+
+ tmp = AVI_read_frame(
+ anim->avi, AVI_FORMAT_RGB32, position, AVI_get_stream(anim->avi, AVIST_VIDEO, 0));
+
+ if (tmp == NULL) {
+ printf("Error reading frame from AVI: '%s'\n", anim->name);
+ IMB_freeImBuf(ibuf);
+ return NULL;
+ }
+
+ for (y = 0; y < anim->y; y++) {
+ memcpy(&(ibuf->rect)[((anim->y - y) - 1) * anim->x], &tmp[y * anim->x], anim->x * 4);
+ }
+
+ MEM_freeN(tmp);
+ }
+
+ ibuf->rect_colorspace = colormanage_colorspace_get_named(anim->colorspace);
+
+ return ibuf;
}
-#endif /* WITH_AVI */
+#endif /* WITH_AVI */
#ifdef WITH_FFMPEG
BLI_INLINE bool need_aligned_ffmpeg_buffer(struct anim *anim)
{
- return (anim->x & 31) != 0;
+ return (anim->x & 31) != 0;
}
static int startffmpeg(struct anim *anim)
{
- int i, videoStream;
-
- AVCodec *pCodec;
- AVFormatContext *pFormatCtx = NULL;
- AVCodecContext *pCodecCtx;
- AVRational frame_rate;
- int frs_num;
- double frs_den;
- int streamcount;
-
-#ifdef FFMPEG_SWSCALE_COLOR_SPACE_SUPPORT
- /* The following for color space determination */
- int srcRange, dstRange, brightness, contrast, saturation;
- int *table;
- const int *inv_table;
-#endif
-
- if (anim == NULL) return(-1);
-
- streamcount = anim->streamindex;
-
- if (avformat_open_input(&pFormatCtx, anim->name, NULL, NULL) != 0) {
- return -1;
- }
-
- if (avformat_find_stream_info(pFormatCtx, NULL) < 0) {
- avformat_close_input(&pFormatCtx);
- return -1;
- }
-
- av_dump_format(pFormatCtx, 0, anim->name, 0);
-
-
- /* Find the video stream */
- videoStream = -1;
-
- for (i = 0; i < pFormatCtx->nb_streams; i++)
- if (pFormatCtx->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
- if (streamcount > 0) {
- streamcount--;
- continue;
- }
- videoStream = i;
- break;
- }
-
- if (videoStream == -1) {
- avformat_close_input(&pFormatCtx);
- return -1;
- }
-
- pCodecCtx = pFormatCtx->streams[videoStream]->codec;
-
- /* Find the decoder for the video stream */
- pCodec = avcodec_find_decoder(pCodecCtx->codec_id);
- if (pCodec == NULL) {
- avformat_close_input(&pFormatCtx);
- return -1;
- }
-
- pCodecCtx->workaround_bugs = 1;
-
- if (avcodec_open2(pCodecCtx, pCodec, NULL) < 0) {
- avformat_close_input(&pFormatCtx);
- return -1;
- }
- if (pCodecCtx->pix_fmt == AV_PIX_FMT_NONE) {
- avcodec_close(anim->pCodecCtx);
- avformat_close_input(&pFormatCtx);
- return -1;
- }
-
- frame_rate = av_get_r_frame_rate_compat(pFormatCtx, pFormatCtx->streams[videoStream]);
- if (pFormatCtx->streams[videoStream]->nb_frames != 0) {
- anim->duration = pFormatCtx->streams[videoStream]->nb_frames;
- }
- else {
- anim->duration = (int)(pFormatCtx->duration *
- av_q2d(frame_rate) /
- AV_TIME_BASE + 0.5f);
- }
-
- frs_num = frame_rate.num;
- frs_den = frame_rate.den;
-
- frs_den *= AV_TIME_BASE;
-
- while (frs_num % 10 == 0 && frs_den >= 2.0 && frs_num > 10) {
- frs_num /= 10;
- frs_den /= 10;
- }
-
- anim->frs_sec = frs_num;
- anim->frs_sec_base = frs_den;
-
- anim->params = 0;
-
- anim->x = pCodecCtx->width;
- anim->y = av_get_cropped_height_from_codec(pCodecCtx);
-
- anim->pFormatCtx = pFormatCtx;
- anim->pCodecCtx = pCodecCtx;
- anim->pCodec = pCodec;
- anim->videoStream = videoStream;
-
- anim->interlacing = 0;
- anim->orientation = 0;
- anim->framesize = anim->x * anim->y * 4;
-
- anim->curposition = -1;
- anim->last_frame = 0;
- anim->last_pts = -1;
- anim->next_pts = -1;
- anim->next_packet.stream_index = -1;
-
- anim->pFrame = av_frame_alloc();
- anim->pFrameComplete = false;
- anim->pFrameDeinterlaced = av_frame_alloc();
- anim->pFrameRGB = av_frame_alloc();
-
- if (need_aligned_ffmpeg_buffer(anim)) {
- anim->pFrameRGB->format = AV_PIX_FMT_RGBA;
- anim->pFrameRGB->width = anim->x;
- anim->pFrameRGB->height = anim->y;
-
- if (av_frame_get_buffer(anim->pFrameRGB, 32) < 0) {
- fprintf(stderr, "Could not allocate frame data.\n");
- avcodec_close(anim->pCodecCtx);
- avformat_close_input(&anim->pFormatCtx);
- av_frame_free(&anim->pFrameRGB);
- av_frame_free(&anim->pFrameDeinterlaced);
- av_frame_free(&anim->pFrame);
- anim->pCodecCtx = NULL;
- return -1;
- }
- }
-
- if (avpicture_get_size(AV_PIX_FMT_RGBA, anim->x, anim->y) !=
- anim->x * anim->y * 4)
- {
- fprintf(stderr,
- "ffmpeg has changed alloc scheme ... ARGHHH!\n");
- avcodec_close(anim->pCodecCtx);
- avformat_close_input(&anim->pFormatCtx);
- av_frame_free(&anim->pFrameRGB);
- av_frame_free(&anim->pFrameDeinterlaced);
- av_frame_free(&anim->pFrame);
- anim->pCodecCtx = NULL;
- return -1;
- }
-
- if (anim->ib_flags & IB_animdeinterlace) {
- avpicture_fill((AVPicture *) anim->pFrameDeinterlaced,
- MEM_callocN(avpicture_get_size(
- anim->pCodecCtx->pix_fmt,
- anim->pCodecCtx->width,
- anim->pCodecCtx->height),
- "ffmpeg deinterlace"),
- anim->pCodecCtx->pix_fmt,
- anim->pCodecCtx->width,
- anim->pCodecCtx->height);
- }
-
- if (pCodecCtx->has_b_frames) {
- anim->preseek = 25; /* FIXME: detect gopsize ... */
- }
- else {
- anim->preseek = 0;
- }
-
- anim->img_convert_ctx = sws_getContext(
- anim->x,
- anim->y,
- anim->pCodecCtx->pix_fmt,
- anim->x,
- anim->y,
- AV_PIX_FMT_RGBA,
- SWS_FAST_BILINEAR | SWS_PRINT_INFO | SWS_FULL_CHR_H_INT,
- NULL, NULL, NULL);
-
- if (!anim->img_convert_ctx) {
- fprintf(stderr,
- "Can't transform color space??? Bailing out...\n");
- avcodec_close(anim->pCodecCtx);
- avformat_close_input(&anim->pFormatCtx);
- av_frame_free(&anim->pFrameRGB);
- av_frame_free(&anim->pFrameDeinterlaced);
- av_frame_free(&anim->pFrame);
- anim->pCodecCtx = NULL;
- return -1;
- }
-
-#ifdef FFMPEG_SWSCALE_COLOR_SPACE_SUPPORT
- /* Try do detect if input has 0-255 YCbCR range (JFIF Jpeg MotionJpeg) */
- if (!sws_getColorspaceDetails(anim->img_convert_ctx, (int **)&inv_table, &srcRange,
- &table, &dstRange, &brightness, &contrast, &saturation))
- {
- srcRange = srcRange || anim->pCodecCtx->color_range == AVCOL_RANGE_JPEG;
- inv_table = sws_getCoefficients(anim->pCodecCtx->colorspace);
-
- if (sws_setColorspaceDetails(anim->img_convert_ctx, (int *)inv_table, srcRange,
- table, dstRange, brightness, contrast, saturation))
- {
- fprintf(stderr, "Warning: Could not set libswscale colorspace details.\n");
- }
- }
- else {
- fprintf(stderr, "Warning: Could not set libswscale colorspace details.\n");
- }
-#endif
-
- return (0);
+ int i, videoStream;
+
+ AVCodec *pCodec;
+ AVFormatContext *pFormatCtx = NULL;
+ AVCodecContext *pCodecCtx;
+ AVRational frame_rate;
+ int frs_num;
+ double frs_den;
+ int streamcount;
+
+# ifdef FFMPEG_SWSCALE_COLOR_SPACE_SUPPORT
+ /* The following for color space determination */
+ int srcRange, dstRange, brightness, contrast, saturation;
+ int *table;
+ const int *inv_table;
+# endif
+
+ if (anim == NULL)
+ return (-1);
+
+ streamcount = anim->streamindex;
+
+ if (avformat_open_input(&pFormatCtx, anim->name, NULL, NULL) != 0) {
+ return -1;
+ }
+
+ if (avformat_find_stream_info(pFormatCtx, NULL) < 0) {
+ avformat_close_input(&pFormatCtx);
+ return -1;
+ }
+
+ av_dump_format(pFormatCtx, 0, anim->name, 0);
+
+ /* Find the video stream */
+ videoStream = -1;
+
+ for (i = 0; i < pFormatCtx->nb_streams; i++)
+ if (pFormatCtx->streams[i]->codec->codec_type == AVMEDIA_TYPE_VIDEO) {
+ if (streamcount > 0) {
+ streamcount--;
+ continue;
+ }
+ videoStream = i;
+ break;
+ }
+
+ if (videoStream == -1) {
+ avformat_close_input(&pFormatCtx);
+ return -1;
+ }
+
+ pCodecCtx = pFormatCtx->streams[videoStream]->codec;
+
+ /* Find the decoder for the video stream */
+ pCodec = avcodec_find_decoder(pCodecCtx->codec_id);
+ if (pCodec == NULL) {
+ avformat_close_input(&pFormatCtx);
+ return -1;
+ }
+
+ pCodecCtx->workaround_bugs = 1;
+
+ if (avcodec_open2(pCodecCtx, pCodec, NULL) < 0) {
+ avformat_close_input(&pFormatCtx);
+ return -1;
+ }
+ if (pCodecCtx->pix_fmt == AV_PIX_FMT_NONE) {
+ avcodec_close(anim->pCodecCtx);
+ avformat_close_input(&pFormatCtx);
+ return -1;
+ }
+
+ frame_rate = av_get_r_frame_rate_compat(pFormatCtx, pFormatCtx->streams[videoStream]);
+ if (pFormatCtx->streams[videoStream]->nb_frames != 0) {
+ anim->duration = pFormatCtx->streams[videoStream]->nb_frames;
+ }
+ else {
+ anim->duration = (int)(pFormatCtx->duration * av_q2d(frame_rate) / AV_TIME_BASE + 0.5f);
+ }
+
+ frs_num = frame_rate.num;
+ frs_den = frame_rate.den;
+
+ frs_den *= AV_TIME_BASE;
+
+ while (frs_num % 10 == 0 && frs_den >= 2.0 && frs_num > 10) {
+ frs_num /= 10;
+ frs_den /= 10;
+ }
+
+ anim->frs_sec = frs_num;
+ anim->frs_sec_base = frs_den;
+
+ anim->params = 0;
+
+ anim->x = pCodecCtx->width;
+ anim->y = av_get_cropped_height_from_codec(pCodecCtx);
+
+ anim->pFormatCtx = pFormatCtx;
+ anim->pCodecCtx = pCodecCtx;
+ anim->pCodec = pCodec;
+ anim->videoStream = videoStream;
+
+ anim->interlacing = 0;
+ anim->orientation = 0;
+ anim->framesize = anim->x * anim->y * 4;
+
+ anim->curposition = -1;
+ anim->last_frame = 0;
+ anim->last_pts = -1;
+ anim->next_pts = -1;
+ anim->next_packet.stream_index = -1;
+
+ anim->pFrame = av_frame_alloc();
+ anim->pFrameComplete = false;
+ anim->pFrameDeinterlaced = av_frame_alloc();
+ anim->pFrameRGB = av_frame_alloc();
+
+ if (need_aligned_ffmpeg_buffer(anim)) {
+ anim->pFrameRGB->format = AV_PIX_FMT_RGBA;
+ anim->pFrameRGB->width = anim->x;
+ anim->pFrameRGB->height = anim->y;
+
+ if (av_frame_get_buffer(anim->pFrameRGB, 32) < 0) {
+ fprintf(stderr, "Could not allocate frame data.\n");
+ avcodec_close(anim->pCodecCtx);
+ avformat_close_input(&anim->pFormatCtx);
+ av_frame_free(&anim->pFrameRGB);
+ av_frame_free(&anim->pFrameDeinterlaced);
+ av_frame_free(&anim->pFrame);
+ anim->pCodecCtx = NULL;
+ return -1;
+ }
+ }
+
+ if (avpicture_get_size(AV_PIX_FMT_RGBA, anim->x, anim->y) != anim->x * anim->y * 4) {
+ fprintf(stderr, "ffmpeg has changed alloc scheme ... ARGHHH!\n");
+ avcodec_close(anim->pCodecCtx);
+ avformat_close_input(&anim->pFormatCtx);
+ av_frame_free(&anim->pFrameRGB);
+ av_frame_free(&anim->pFrameDeinterlaced);
+ av_frame_free(&anim->pFrame);
+ anim->pCodecCtx = NULL;
+ return -1;
+ }
+
+ if (anim->ib_flags & IB_animdeinterlace) {
+ avpicture_fill((AVPicture *)anim->pFrameDeinterlaced,
+ MEM_callocN(avpicture_get_size(anim->pCodecCtx->pix_fmt,
+ anim->pCodecCtx->width,
+ anim->pCodecCtx->height),
+ "ffmpeg deinterlace"),
+ anim->pCodecCtx->pix_fmt,
+ anim->pCodecCtx->width,
+ anim->pCodecCtx->height);
+ }
+
+ if (pCodecCtx->has_b_frames) {
+ anim->preseek = 25; /* FIXME: detect gopsize ... */
+ }
+ else {
+ anim->preseek = 0;
+ }
+
+ anim->img_convert_ctx = sws_getContext(anim->x,
+ anim->y,
+ anim->pCodecCtx->pix_fmt,
+ anim->x,
+ anim->y,
+ AV_PIX_FMT_RGBA,
+ SWS_FAST_BILINEAR | SWS_PRINT_INFO | SWS_FULL_CHR_H_INT,
+ NULL,
+ NULL,
+ NULL);
+
+ if (!anim->img_convert_ctx) {
+ fprintf(stderr, "Can't transform color space??? Bailing out...\n");
+ avcodec_close(anim->pCodecCtx);
+ avformat_close_input(&anim->pFormatCtx);
+ av_frame_free(&anim->pFrameRGB);
+ av_frame_free(&anim->pFrameDeinterlaced);
+ av_frame_free(&anim->pFrame);
+ anim->pCodecCtx = NULL;
+ return -1;
+ }
+
+# ifdef FFMPEG_SWSCALE_COLOR_SPACE_SUPPORT
+ /* Try do detect if input has 0-255 YCbCR range (JFIF Jpeg MotionJpeg) */
+ if (!sws_getColorspaceDetails(anim->img_convert_ctx,
+ (int **)&inv_table,
+ &srcRange,
+ &table,
+ &dstRange,
+ &brightness,
+ &contrast,
+ &saturation)) {
+ srcRange = srcRange || anim->pCodecCtx->color_range == AVCOL_RANGE_JPEG;
+ inv_table = sws_getCoefficients(anim->pCodecCtx->colorspace);
+
+ if (sws_setColorspaceDetails(anim->img_convert_ctx,
+ (int *)inv_table,
+ srcRange,
+ table,
+ dstRange,
+ brightness,
+ contrast,
+ saturation)) {
+ fprintf(stderr, "Warning: Could not set libswscale colorspace details.\n");
+ }
+ }
+ else {
+ fprintf(stderr, "Warning: Could not set libswscale colorspace details.\n");
+ }
+# endif
+
+ return (0);
}
/* postprocess the image in anim->pFrame and do color conversion
@@ -689,535 +705,504 @@ static int startffmpeg(struct anim *anim)
static void ffmpeg_postprocess(struct anim *anim)
{
- AVFrame *input = anim->pFrame;
- ImBuf *ibuf = anim->last_frame;
- int filter_y = 0;
-
- if (!anim->pFrameComplete) {
- return;
- }
-
- /* This means the data wasnt read properly,
- * this check stops crashing */
- if (input->data[0] == 0 && input->data[1] == 0 &&
- input->data[2] == 0 && input->data[3] == 0)
- {
- fprintf(stderr, "ffmpeg_fetchibuf: "
- "data not read properly...\n");
- return;
- }
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- " POSTPROC: anim->pFrame planes: %p %p %p %p\n",
- input->data[0], input->data[1], input->data[2],
- input->data[3]);
-
-
- if (anim->ib_flags & IB_animdeinterlace) {
- if (avpicture_deinterlace(
- (AVPicture *)
- anim->pFrameDeinterlaced,
- (const AVPicture *)
- anim->pFrame,
- anim->pCodecCtx->pix_fmt,
- anim->pCodecCtx->width,
- anim->pCodecCtx->height) < 0)
- {
- filter_y = true;
- }
- else {
- input = anim->pFrameDeinterlaced;
- }
- }
-
- if (!need_aligned_ffmpeg_buffer(anim)) {
- avpicture_fill((AVPicture *) anim->pFrameRGB,
- (unsigned char *) ibuf->rect,
- AV_PIX_FMT_RGBA, anim->x, anim->y);
- }
-
- if (ENDIAN_ORDER == B_ENDIAN) {
- int *dstStride = anim->pFrameRGB->linesize;
- uint8_t **dst = anim->pFrameRGB->data;
- int dstStride2[4] = { dstStride[0], 0, 0, 0 };
- uint8_t *dst2[4] = { dst[0], 0, 0, 0 };
- int x, y, h, w;
- unsigned char *bottom;
- unsigned char *top;
-
- sws_scale(anim->img_convert_ctx,
- (const uint8_t *const *)input->data,
- input->linesize,
- 0,
- anim->y,
- dst2,
- dstStride2);
-
- bottom = (unsigned char *) ibuf->rect;
- top = bottom + ibuf->x * (ibuf->y - 1) * 4;
-
- h = (ibuf->y + 1) / 2;
- w = ibuf->x;
-
- for (y = 0; y < h; y++) {
- unsigned char tmp[4];
- unsigned int *tmp_l =
- (unsigned int *) tmp;
-
- for (x = 0; x < w; x++) {
- tmp[0] = bottom[0];
- tmp[1] = bottom[1];
- tmp[2] = bottom[2];
- tmp[3] = bottom[3];
-
- bottom[0] = top[0];
- bottom[1] = top[1];
- bottom[2] = top[2];
- bottom[3] = top[3];
-
- *(unsigned int *) top = *tmp_l;
-
- bottom += 4;
- top += 4;
- }
- top -= 8 * w;
- }
- }
- else {
- int *dstStride = anim->pFrameRGB->linesize;
- uint8_t **dst = anim->pFrameRGB->data;
- int dstStride2[4] = { -dstStride[0], 0, 0, 0 };
- uint8_t *dst2[4] = { dst[0] + (anim->y - 1) * dstStride[0], 0, 0, 0 };
-
- sws_scale(anim->img_convert_ctx,
- (const uint8_t *const *)input->data,
- input->linesize,
- 0,
- anim->y,
- dst2,
- dstStride2);
- }
-
- if (need_aligned_ffmpeg_buffer(anim)) {
- uint8_t *src = anim->pFrameRGB->data[0];
- uint8_t *dst = (uint8_t *) ibuf->rect;
- for (int y = 0; y < anim->y; y++) {
- memcpy(dst, src, anim->x * 4);
- dst += anim->x * 4;
- src += anim->pFrameRGB->linesize[0];
- }
- }
-
- if (filter_y) {
- IMB_filtery(ibuf);
- }
+ AVFrame *input = anim->pFrame;
+ ImBuf *ibuf = anim->last_frame;
+ int filter_y = 0;
+
+ if (!anim->pFrameComplete) {
+ return;
+ }
+
+ /* This means the data wasnt read properly,
+ * this check stops crashing */
+ if (input->data[0] == 0 && input->data[1] == 0 && input->data[2] == 0 && input->data[3] == 0) {
+ fprintf(stderr,
+ "ffmpeg_fetchibuf: "
+ "data not read properly...\n");
+ return;
+ }
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ " POSTPROC: anim->pFrame planes: %p %p %p %p\n",
+ input->data[0],
+ input->data[1],
+ input->data[2],
+ input->data[3]);
+
+ if (anim->ib_flags & IB_animdeinterlace) {
+ if (avpicture_deinterlace((AVPicture *)anim->pFrameDeinterlaced,
+ (const AVPicture *)anim->pFrame,
+ anim->pCodecCtx->pix_fmt,
+ anim->pCodecCtx->width,
+ anim->pCodecCtx->height) < 0) {
+ filter_y = true;
+ }
+ else {
+ input = anim->pFrameDeinterlaced;
+ }
+ }
+
+ if (!need_aligned_ffmpeg_buffer(anim)) {
+ avpicture_fill((AVPicture *)anim->pFrameRGB,
+ (unsigned char *)ibuf->rect,
+ AV_PIX_FMT_RGBA,
+ anim->x,
+ anim->y);
+ }
+
+ if (ENDIAN_ORDER == B_ENDIAN) {
+ int *dstStride = anim->pFrameRGB->linesize;
+ uint8_t **dst = anim->pFrameRGB->data;
+ int dstStride2[4] = {dstStride[0], 0, 0, 0};
+ uint8_t *dst2[4] = {dst[0], 0, 0, 0};
+ int x, y, h, w;
+ unsigned char *bottom;
+ unsigned char *top;
+
+ sws_scale(anim->img_convert_ctx,
+ (const uint8_t *const *)input->data,
+ input->linesize,
+ 0,
+ anim->y,
+ dst2,
+ dstStride2);
+
+ bottom = (unsigned char *)ibuf->rect;
+ top = bottom + ibuf->x * (ibuf->y - 1) * 4;
+
+ h = (ibuf->y + 1) / 2;
+ w = ibuf->x;
+
+ for (y = 0; y < h; y++) {
+ unsigned char tmp[4];
+ unsigned int *tmp_l = (unsigned int *)tmp;
+
+ for (x = 0; x < w; x++) {
+ tmp[0] = bottom[0];
+ tmp[1] = bottom[1];
+ tmp[2] = bottom[2];
+ tmp[3] = bottom[3];
+
+ bottom[0] = top[0];
+ bottom[1] = top[1];
+ bottom[2] = top[2];
+ bottom[3] = top[3];
+
+ *(unsigned int *)top = *tmp_l;
+
+ bottom += 4;
+ top += 4;
+ }
+ top -= 8 * w;
+ }
+ }
+ else {
+ int *dstStride = anim->pFrameRGB->linesize;
+ uint8_t **dst = anim->pFrameRGB->data;
+ int dstStride2[4] = {-dstStride[0], 0, 0, 0};
+ uint8_t *dst2[4] = {dst[0] + (anim->y - 1) * dstStride[0], 0, 0, 0};
+
+ sws_scale(anim->img_convert_ctx,
+ (const uint8_t *const *)input->data,
+ input->linesize,
+ 0,
+ anim->y,
+ dst2,
+ dstStride2);
+ }
+
+ if (need_aligned_ffmpeg_buffer(anim)) {
+ uint8_t *src = anim->pFrameRGB->data[0];
+ uint8_t *dst = (uint8_t *)ibuf->rect;
+ for (int y = 0; y < anim->y; y++) {
+ memcpy(dst, src, anim->x * 4);
+ dst += anim->x * 4;
+ src += anim->pFrameRGB->linesize[0];
+ }
+ }
+
+ if (filter_y) {
+ IMB_filtery(ibuf);
+ }
}
/* decode one video frame also considering the packet read into next_packet */
static int ffmpeg_decode_video_frame(struct anim *anim)
{
- int rval = 0;
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG, " DECODE VIDEO FRAME\n");
-
- if (anim->next_packet.stream_index == anim->videoStream) {
- av_free_packet(&anim->next_packet);
- anim->next_packet.stream_index = -1;
- }
-
- while ((rval = av_read_frame(anim->pFormatCtx, &anim->next_packet)) >= 0) {
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG,
- "%sREAD: strID=%d (VID: %d) dts=%lld pts=%lld "
- "%s\n",
- (anim->next_packet.stream_index == anim->videoStream)
- ? "->" : " ",
- anim->next_packet.stream_index,
- anim->videoStream,
- (anim->next_packet.dts == AV_NOPTS_VALUE) ? -1 :
- (long long int)anim->next_packet.dts,
- (anim->next_packet.pts == AV_NOPTS_VALUE) ? -1 :
- (long long int)anim->next_packet.pts,
- (anim->next_packet.flags & AV_PKT_FLAG_KEY) ?
- " KEY" : "");
- if (anim->next_packet.stream_index == anim->videoStream) {
- anim->pFrameComplete = 0;
-
- avcodec_decode_video2(
- anim->pCodecCtx,
- anim->pFrame, &anim->pFrameComplete,
- &anim->next_packet);
-
- if (anim->pFrameComplete) {
- anim->next_pts = av_get_pts_from_frame(
- anim->pFormatCtx, anim->pFrame);
-
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG,
- " FRAME DONE: next_pts=%lld "
- "pkt_pts=%lld, guessed_pts=%lld\n",
- (anim->pFrame->pts == AV_NOPTS_VALUE) ?
- -1 : (long long int)anim->pFrame->pts,
- (anim->pFrame->pkt_pts == AV_NOPTS_VALUE) ?
- -1 : (long long int)anim->pFrame->pkt_pts,
- (long long int)anim->next_pts);
- break;
- }
- }
- av_free_packet(&anim->next_packet);
- anim->next_packet.stream_index = -1;
- }
-
- if (rval == AVERROR_EOF) {
- /* this sets size and data fields to zero,
- * which is necessary to decode the remaining data
- * in the decoder engine after EOF. It also prevents a memory
- * leak, since av_read_frame spills out a full size packet even
- * on EOF... (and: it's safe to call on NULL packets) */
-
- av_free_packet(&anim->next_packet);
-
- anim->next_packet.size = 0;
- anim->next_packet.data = 0;
-
- anim->pFrameComplete = 0;
-
- avcodec_decode_video2(
- anim->pCodecCtx,
- anim->pFrame, &anim->pFrameComplete,
- &anim->next_packet);
-
- if (anim->pFrameComplete) {
- anim->next_pts = av_get_pts_from_frame(
- anim->pFormatCtx, anim->pFrame);
-
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG,
- " FRAME DONE (after EOF): next_pts=%lld "
- "pkt_pts=%lld, guessed_pts=%lld\n",
- (anim->pFrame->pts == AV_NOPTS_VALUE) ?
- -1 : (long long int)anim->pFrame->pts,
- (anim->pFrame->pkt_pts == AV_NOPTS_VALUE) ?
- -1 : (long long int)anim->pFrame->pkt_pts,
- (long long int)anim->next_pts);
- rval = 0;
- }
- }
-
- if (rval < 0) {
- anim->next_packet.stream_index = -1;
-
- av_log(anim->pFormatCtx,
- AV_LOG_ERROR, " DECODE READ FAILED: av_read_frame() "
- "returned error: %d\n", rval);
- }
-
- return (rval >= 0);
+ int rval = 0;
+
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, " DECODE VIDEO FRAME\n");
+
+ if (anim->next_packet.stream_index == anim->videoStream) {
+ av_free_packet(&anim->next_packet);
+ anim->next_packet.stream_index = -1;
+ }
+
+ while ((rval = av_read_frame(anim->pFormatCtx, &anim->next_packet)) >= 0) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "%sREAD: strID=%d (VID: %d) dts=%lld pts=%lld "
+ "%s\n",
+ (anim->next_packet.stream_index == anim->videoStream) ? "->" : " ",
+ anim->next_packet.stream_index,
+ anim->videoStream,
+ (anim->next_packet.dts == AV_NOPTS_VALUE) ? -1 : (long long int)anim->next_packet.dts,
+ (anim->next_packet.pts == AV_NOPTS_VALUE) ? -1 : (long long int)anim->next_packet.pts,
+ (anim->next_packet.flags & AV_PKT_FLAG_KEY) ? " KEY" : "");
+ if (anim->next_packet.stream_index == anim->videoStream) {
+ anim->pFrameComplete = 0;
+
+ avcodec_decode_video2(
+ anim->pCodecCtx, anim->pFrame, &anim->pFrameComplete, &anim->next_packet);
+
+ if (anim->pFrameComplete) {
+ anim->next_pts = av_get_pts_from_frame(anim->pFormatCtx, anim->pFrame);
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ " FRAME DONE: next_pts=%lld "
+ "pkt_pts=%lld, guessed_pts=%lld\n",
+ (anim->pFrame->pts == AV_NOPTS_VALUE) ? -1 : (long long int)anim->pFrame->pts,
+ (anim->pFrame->pkt_pts == AV_NOPTS_VALUE) ? -1 :
+ (long long int)anim->pFrame->pkt_pts,
+ (long long int)anim->next_pts);
+ break;
+ }
+ }
+ av_free_packet(&anim->next_packet);
+ anim->next_packet.stream_index = -1;
+ }
+
+ if (rval == AVERROR_EOF) {
+ /* this sets size and data fields to zero,
+ * which is necessary to decode the remaining data
+ * in the decoder engine after EOF. It also prevents a memory
+ * leak, since av_read_frame spills out a full size packet even
+ * on EOF... (and: it's safe to call on NULL packets) */
+
+ av_free_packet(&anim->next_packet);
+
+ anim->next_packet.size = 0;
+ anim->next_packet.data = 0;
+
+ anim->pFrameComplete = 0;
+
+ avcodec_decode_video2(
+ anim->pCodecCtx, anim->pFrame, &anim->pFrameComplete, &anim->next_packet);
+
+ if (anim->pFrameComplete) {
+ anim->next_pts = av_get_pts_from_frame(anim->pFormatCtx, anim->pFrame);
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ " FRAME DONE (after EOF): next_pts=%lld "
+ "pkt_pts=%lld, guessed_pts=%lld\n",
+ (anim->pFrame->pts == AV_NOPTS_VALUE) ? -1 : (long long int)anim->pFrame->pts,
+ (anim->pFrame->pkt_pts == AV_NOPTS_VALUE) ? -1 : (long long int)anim->pFrame->pkt_pts,
+ (long long int)anim->next_pts);
+ rval = 0;
+ }
+ }
+
+ if (rval < 0) {
+ anim->next_packet.stream_index = -1;
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_ERROR,
+ " DECODE READ FAILED: av_read_frame() "
+ "returned error: %d\n",
+ rval);
+ }
+
+ return (rval >= 0);
}
-static void ffmpeg_decode_video_frame_scan(
- struct anim *anim, int64_t pts_to_search)
+static void ffmpeg_decode_video_frame_scan(struct anim *anim, int64_t pts_to_search)
{
- /* there seem to exist *very* silly GOP lengths out in the wild... */
- int count = 1000;
-
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG,
- "SCAN start: considering pts=%lld in search of %lld\n",
- (long long int)anim->next_pts, (long long int)pts_to_search);
-
- while (count > 0 && anim->next_pts < pts_to_search) {
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG,
- " WHILE: pts=%lld in search of %lld\n",
- (long long int)anim->next_pts, (long long int)pts_to_search);
- if (!ffmpeg_decode_video_frame(anim)) {
- break;
- }
- count--;
- }
- if (count == 0) {
- av_log(anim->pFormatCtx,
- AV_LOG_ERROR,
- "SCAN failed: completely lost in stream, "
- "bailing out at PTS=%lld, searching for PTS=%lld\n",
- (long long int)anim->next_pts, (long long int)pts_to_search);
- }
- if (anim->next_pts == pts_to_search) {
- av_log(anim->pFormatCtx,
- AV_LOG_DEBUG, "SCAN HAPPY: we found our PTS!\n");
- }
- else {
- av_log(anim->pFormatCtx,
- AV_LOG_ERROR, "SCAN UNHAPPY: PTS not matched!\n");
- }
+ /* there seem to exist *very* silly GOP lengths out in the wild... */
+ int count = 1000;
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "SCAN start: considering pts=%lld in search of %lld\n",
+ (long long int)anim->next_pts,
+ (long long int)pts_to_search);
+
+ while (count > 0 && anim->next_pts < pts_to_search) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ " WHILE: pts=%lld in search of %lld\n",
+ (long long int)anim->next_pts,
+ (long long int)pts_to_search);
+ if (!ffmpeg_decode_video_frame(anim)) {
+ break;
+ }
+ count--;
+ }
+ if (count == 0) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_ERROR,
+ "SCAN failed: completely lost in stream, "
+ "bailing out at PTS=%lld, searching for PTS=%lld\n",
+ (long long int)anim->next_pts,
+ (long long int)pts_to_search);
+ }
+ if (anim->next_pts == pts_to_search) {
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "SCAN HAPPY: we found our PTS!\n");
+ }
+ else {
+ av_log(anim->pFormatCtx, AV_LOG_ERROR, "SCAN UNHAPPY: PTS not matched!\n");
+ }
}
static int match_format(const char *name, AVFormatContext *pFormatCtx)
{
- const char *p;
- int len, namelen;
-
- const char *names = pFormatCtx->iformat->name;
-
- if (!name || !names)
- return 0;
-
- namelen = strlen(name);
- while ((p = strchr(names, ','))) {
- len = MAX2(p - names, namelen);
- if (!BLI_strncasecmp(name, names, len))
- return 1;
- names = p + 1;
- }
- return !BLI_strcasecmp(name, names);
+ const char *p;
+ int len, namelen;
+
+ const char *names = pFormatCtx->iformat->name;
+
+ if (!name || !names)
+ return 0;
+
+ namelen = strlen(name);
+ while ((p = strchr(names, ','))) {
+ len = MAX2(p - names, namelen);
+ if (!BLI_strncasecmp(name, names, len))
+ return 1;
+ names = p + 1;
+ }
+ return !BLI_strcasecmp(name, names);
}
static int ffmpeg_seek_by_byte(AVFormatContext *pFormatCtx)
{
- static const char *byte_seek_list[] = { "mpegts", 0 };
- const char **p;
+ static const char *byte_seek_list[] = {"mpegts", 0};
+ const char **p;
- if (pFormatCtx->iformat->flags & AVFMT_TS_DISCONT) {
- return true;
- }
+ if (pFormatCtx->iformat->flags & AVFMT_TS_DISCONT) {
+ return true;
+ }
- p = byte_seek_list;
+ p = byte_seek_list;
- while (*p) {
- if (match_format(*p++, pFormatCtx)) {
- return true;
- }
- }
+ while (*p) {
+ if (match_format(*p++, pFormatCtx)) {
+ return true;
+ }
+ }
- return false;
+ return false;
}
-static ImBuf *ffmpeg_fetchibuf(struct anim *anim, int position,
- IMB_Timecode_Type tc)
+static ImBuf *ffmpeg_fetchibuf(struct anim *anim, int position, IMB_Timecode_Type tc)
{
- int64_t pts_to_search = 0;
- double frame_rate;
- double pts_time_base;
- long long st_time;
- struct anim_index *tc_index = 0;
- AVStream *v_st;
- int new_frame_index = 0; /* To quiet gcc barking... */
- int old_frame_index = 0; /* To quiet gcc barking... */
-
- if (anim == NULL) return (0);
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG, "FETCH: pos=%d\n", position);
-
- if (tc != IMB_TC_NONE) {
- tc_index = IMB_anim_open_index(anim, tc);
- }
-
- v_st = anim->pFormatCtx->streams[anim->videoStream];
-
- frame_rate = av_q2d(av_get_r_frame_rate_compat(anim->pFormatCtx, v_st));
-
- st_time = anim->pFormatCtx->start_time;
- pts_time_base = av_q2d(v_st->time_base);
-
- if (tc_index) {
- new_frame_index = IMB_indexer_get_frame_index(
- tc_index, position);
- old_frame_index = IMB_indexer_get_frame_index(
- tc_index, anim->curposition);
- pts_to_search = IMB_indexer_get_pts(
- tc_index, new_frame_index);
- }
- else {
- pts_to_search = (long long)
- floor(((double) position) /
- pts_time_base / frame_rate + 0.5);
-
- if (st_time != AV_NOPTS_VALUE) {
- pts_to_search += st_time / pts_time_base / AV_TIME_BASE;
- }
- }
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "FETCH: looking for PTS=%lld "
- "(pts_timebase=%g, frame_rate=%g, st_time=%lld)\n",
- (long long int)pts_to_search, pts_time_base, frame_rate, st_time);
-
- if (anim->last_frame &&
- anim->last_pts <= pts_to_search && anim->next_pts > pts_to_search)
- {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "FETCH: frame repeat: last: %lld next: %lld\n",
- (long long int)anim->last_pts,
- (long long int)anim->next_pts);
- IMB_refImBuf(anim->last_frame);
- anim->curposition = position;
- return anim->last_frame;
- }
-
- if (position > anim->curposition + 1 &&
- anim->preseek &&
- !tc_index &&
- position - (anim->curposition + 1) < anim->preseek)
- {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "FETCH: within preseek interval (no index)\n");
-
- ffmpeg_decode_video_frame_scan(anim, pts_to_search);
- }
- else if (tc_index &&
- IMB_indexer_can_scan(tc_index, old_frame_index,
- new_frame_index))
- {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "FETCH: within preseek interval "
- "(index tells us)\n");
-
- ffmpeg_decode_video_frame_scan(anim, pts_to_search);
- }
- else if (position != anim->curposition + 1) {
- long long pos;
- int ret;
-
- if (tc_index) {
- unsigned long long dts;
-
- pos = IMB_indexer_get_seek_pos(
- tc_index, new_frame_index);
- dts = IMB_indexer_get_seek_pos_dts(
- tc_index, new_frame_index);
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "TC INDEX seek pos = %lld\n", pos);
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "TC INDEX seek dts = %llu\n", dts);
-
- if (ffmpeg_seek_by_byte(anim->pFormatCtx)) {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "... using BYTE pos\n");
-
- ret = av_seek_frame(anim->pFormatCtx,
- -1,
- pos, AVSEEK_FLAG_BYTE);
- av_update_cur_dts(anim->pFormatCtx, v_st, dts);
- }
- else {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "... using DTS pos\n");
- ret = av_seek_frame(anim->pFormatCtx,
- anim->videoStream,
- dts, AVSEEK_FLAG_BACKWARD);
- }
- }
- else {
- pos = (long long) (position - anim->preseek) *
- AV_TIME_BASE / frame_rate;
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "NO INDEX seek pos = %lld, st_time = %lld\n",
- pos, (st_time != AV_NOPTS_VALUE) ? st_time : 0);
-
- if (pos < 0) {
- pos = 0;
- }
-
- if (st_time != AV_NOPTS_VALUE) {
- pos += st_time;
- }
-
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "NO INDEX final seek pos = %lld\n", pos);
-
- ret = av_seek_frame(anim->pFormatCtx, -1,
- pos, AVSEEK_FLAG_BACKWARD);
- }
-
- if (ret < 0) {
- av_log(anim->pFormatCtx, AV_LOG_ERROR,
- "FETCH: "
- "error while seeking to DTS = %lld "
- "(frameno = %d, PTS = %lld): errcode = %d\n",
- pos, position, (long long int)pts_to_search, ret);
- }
-
- avcodec_flush_buffers(anim->pCodecCtx);
-
- anim->next_pts = -1;
-
- if (anim->next_packet.stream_index == anim->videoStream) {
- av_free_packet(&anim->next_packet);
- anim->next_packet.stream_index = -1;
- }
-
- /* memset(anim->pFrame, ...) ?? */
-
- if (ret >= 0) {
- ffmpeg_decode_video_frame_scan(anim, pts_to_search);
- }
- }
- else if (position == 0 && anim->curposition == -1) {
- /* first frame without seeking special case... */
- ffmpeg_decode_video_frame(anim);
- }
- else {
- av_log(anim->pFormatCtx, AV_LOG_DEBUG,
- "FETCH: no seek necessary, just continue...\n");
- }
-
- IMB_freeImBuf(anim->last_frame);
- anim->last_frame = IMB_allocImBuf(anim->x, anim->y, 32, IB_rect);
- anim->last_frame->rect_colorspace = colormanage_colorspace_get_named(anim->colorspace);
-
- ffmpeg_postprocess(anim);
-
- anim->last_pts = anim->next_pts;
-
- ffmpeg_decode_video_frame(anim);
-
- anim->curposition = position;
-
- IMB_refImBuf(anim->last_frame);
-
- return anim->last_frame;
+ int64_t pts_to_search = 0;
+ double frame_rate;
+ double pts_time_base;
+ long long st_time;
+ struct anim_index *tc_index = 0;
+ AVStream *v_st;
+ int new_frame_index = 0; /* To quiet gcc barking... */
+ int old_frame_index = 0; /* To quiet gcc barking... */
+
+ if (anim == NULL)
+ return (0);
+
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "FETCH: pos=%d\n", position);
+
+ if (tc != IMB_TC_NONE) {
+ tc_index = IMB_anim_open_index(anim, tc);
+ }
+
+ v_st = anim->pFormatCtx->streams[anim->videoStream];
+
+ frame_rate = av_q2d(av_get_r_frame_rate_compat(anim->pFormatCtx, v_st));
+
+ st_time = anim->pFormatCtx->start_time;
+ pts_time_base = av_q2d(v_st->time_base);
+
+ if (tc_index) {
+ new_frame_index = IMB_indexer_get_frame_index(tc_index, position);
+ old_frame_index = IMB_indexer_get_frame_index(tc_index, anim->curposition);
+ pts_to_search = IMB_indexer_get_pts(tc_index, new_frame_index);
+ }
+ else {
+ pts_to_search = (long long)floor(((double)position) / pts_time_base / frame_rate + 0.5);
+
+ if (st_time != AV_NOPTS_VALUE) {
+ pts_to_search += st_time / pts_time_base / AV_TIME_BASE;
+ }
+ }
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "FETCH: looking for PTS=%lld "
+ "(pts_timebase=%g, frame_rate=%g, st_time=%lld)\n",
+ (long long int)pts_to_search,
+ pts_time_base,
+ frame_rate,
+ st_time);
+
+ if (anim->last_frame && anim->last_pts <= pts_to_search && anim->next_pts > pts_to_search) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "FETCH: frame repeat: last: %lld next: %lld\n",
+ (long long int)anim->last_pts,
+ (long long int)anim->next_pts);
+ IMB_refImBuf(anim->last_frame);
+ anim->curposition = position;
+ return anim->last_frame;
+ }
+
+ if (position > anim->curposition + 1 && anim->preseek && !tc_index &&
+ position - (anim->curposition + 1) < anim->preseek) {
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "FETCH: within preseek interval (no index)\n");
+
+ ffmpeg_decode_video_frame_scan(anim, pts_to_search);
+ }
+ else if (tc_index && IMB_indexer_can_scan(tc_index, old_frame_index, new_frame_index)) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "FETCH: within preseek interval "
+ "(index tells us)\n");
+
+ ffmpeg_decode_video_frame_scan(anim, pts_to_search);
+ }
+ else if (position != anim->curposition + 1) {
+ long long pos;
+ int ret;
+
+ if (tc_index) {
+ unsigned long long dts;
+
+ pos = IMB_indexer_get_seek_pos(tc_index, new_frame_index);
+ dts = IMB_indexer_get_seek_pos_dts(tc_index, new_frame_index);
+
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "TC INDEX seek pos = %lld\n", pos);
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "TC INDEX seek dts = %llu\n", dts);
+
+ if (ffmpeg_seek_by_byte(anim->pFormatCtx)) {
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "... using BYTE pos\n");
+
+ ret = av_seek_frame(anim->pFormatCtx, -1, pos, AVSEEK_FLAG_BYTE);
+ av_update_cur_dts(anim->pFormatCtx, v_st, dts);
+ }
+ else {
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "... using DTS pos\n");
+ ret = av_seek_frame(anim->pFormatCtx, anim->videoStream, dts, AVSEEK_FLAG_BACKWARD);
+ }
+ }
+ else {
+ pos = (long long)(position - anim->preseek) * AV_TIME_BASE / frame_rate;
+
+ av_log(anim->pFormatCtx,
+ AV_LOG_DEBUG,
+ "NO INDEX seek pos = %lld, st_time = %lld\n",
+ pos,
+ (st_time != AV_NOPTS_VALUE) ? st_time : 0);
+
+ if (pos < 0) {
+ pos = 0;
+ }
+
+ if (st_time != AV_NOPTS_VALUE) {
+ pos += st_time;
+ }
+
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "NO INDEX final seek pos = %lld\n", pos);
+
+ ret = av_seek_frame(anim->pFormatCtx, -1, pos, AVSEEK_FLAG_BACKWARD);
+ }
+
+ if (ret < 0) {
+ av_log(anim->pFormatCtx,
+ AV_LOG_ERROR,
+ "FETCH: "
+ "error while seeking to DTS = %lld "
+ "(frameno = %d, PTS = %lld): errcode = %d\n",
+ pos,
+ position,
+ (long long int)pts_to_search,
+ ret);
+ }
+
+ avcodec_flush_buffers(anim->pCodecCtx);
+
+ anim->next_pts = -1;
+
+ if (anim->next_packet.stream_index == anim->videoStream) {
+ av_free_packet(&anim->next_packet);
+ anim->next_packet.stream_index = -1;
+ }
+
+ /* memset(anim->pFrame, ...) ?? */
+
+ if (ret >= 0) {
+ ffmpeg_decode_video_frame_scan(anim, pts_to_search);
+ }
+ }
+ else if (position == 0 && anim->curposition == -1) {
+ /* first frame without seeking special case... */
+ ffmpeg_decode_video_frame(anim);
+ }
+ else {
+ av_log(anim->pFormatCtx, AV_LOG_DEBUG, "FETCH: no seek necessary, just continue...\n");
+ }
+
+ IMB_freeImBuf(anim->last_frame);
+ anim->last_frame = IMB_allocImBuf(anim->x, anim->y, 32, IB_rect);
+ anim->last_frame->rect_colorspace = colormanage_colorspace_get_named(anim->colorspace);
+
+ ffmpeg_postprocess(anim);
+
+ anim->last_pts = anim->next_pts;
+
+ ffmpeg_decode_video_frame(anim);
+
+ anim->curposition = position;
+
+ IMB_refImBuf(anim->last_frame);
+
+ return anim->last_frame;
}
static void free_anim_ffmpeg(struct anim *anim)
{
- if (anim == NULL) return;
-
- if (anim->pCodecCtx) {
- avcodec_close(anim->pCodecCtx);
- avformat_close_input(&anim->pFormatCtx);
-
- /* Special case here: pFrame could share pointers with codec,
- * so in order to avoid double-free we don't use av_frame_free()
- * to free the frame.
- *
- * Could it be a bug in FFmpeg?
- */
- av_free(anim->pFrame);
-
- if (!need_aligned_ffmpeg_buffer(anim)) {
- /* If there's no need for own aligned buffer it means that FFmpeg's
- * frame shares the same buffer as temporary ImBuf. In this case we
- * should not free the buffer when freeing the FFmpeg buffer.
- */
- avpicture_fill((AVPicture *)anim->pFrameRGB,
- NULL,
- AV_PIX_FMT_RGBA,
- anim->x, anim->y);
- }
- av_frame_free(&anim->pFrameRGB);
- av_frame_free(&anim->pFrameDeinterlaced);
-
- sws_freeContext(anim->img_convert_ctx);
- IMB_freeImBuf(anim->last_frame);
- if (anim->next_packet.stream_index != -1) {
- av_free_packet(&anim->next_packet);
- }
- }
- anim->duration = 0;
+ if (anim == NULL)
+ return;
+
+ if (anim->pCodecCtx) {
+ avcodec_close(anim->pCodecCtx);
+ avformat_close_input(&anim->pFormatCtx);
+
+ /* Special case here: pFrame could share pointers with codec,
+ * so in order to avoid double-free we don't use av_frame_free()
+ * to free the frame.
+ *
+ * Could it be a bug in FFmpeg?
+ */
+ av_free(anim->pFrame);
+
+ if (!need_aligned_ffmpeg_buffer(anim)) {
+ /* If there's no need for own aligned buffer it means that FFmpeg's
+ * frame shares the same buffer as temporary ImBuf. In this case we
+ * should not free the buffer when freeing the FFmpeg buffer.
+ */
+ avpicture_fill((AVPicture *)anim->pFrameRGB, NULL, AV_PIX_FMT_RGBA, anim->x, anim->y);
+ }
+ av_frame_free(&anim->pFrameRGB);
+ av_frame_free(&anim->pFrameDeinterlaced);
+
+ sws_freeContext(anim->img_convert_ctx);
+ IMB_freeImBuf(anim->last_frame);
+ if (anim->next_packet.stream_index != -1) {
+ av_free_packet(&anim->next_packet);
+ }
+ }
+ anim->duration = 0;
}
#endif
@@ -1228,208 +1213,211 @@ static void free_anim_ffmpeg(struct anim *anim)
static ImBuf *anim_getnew(struct anim *anim)
{
- struct ImBuf *ibuf = NULL;
+ struct ImBuf *ibuf = NULL;
- if (anim == NULL) return(NULL);
+ if (anim == NULL)
+ return (NULL);
- free_anim_movie(anim);
+ free_anim_movie(anim);
#ifdef WITH_AVI
- free_anim_avi(anim);
+ free_anim_avi(anim);
#endif
#ifdef WITH_FFMPEG
- free_anim_ffmpeg(anim);
+ free_anim_ffmpeg(anim);
#endif
- if (anim->curtype != 0) return (NULL);
- anim->curtype = imb_get_anim_type(anim->name);
-
- switch (anim->curtype) {
- case ANIM_SEQUENCE:
- ibuf = IMB_loadiffname(anim->name, anim->ib_flags, anim->colorspace);
- if (ibuf) {
- BLI_strncpy(anim->first, anim->name, sizeof(anim->first));
- anim->duration = 1;
- }
- break;
- case ANIM_MOVIE:
- if (startmovie(anim)) return (NULL);
- ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0); /* fake */
- break;
+ if (anim->curtype != 0)
+ return (NULL);
+ anim->curtype = imb_get_anim_type(anim->name);
+
+ switch (anim->curtype) {
+ case ANIM_SEQUENCE:
+ ibuf = IMB_loadiffname(anim->name, anim->ib_flags, anim->colorspace);
+ if (ibuf) {
+ BLI_strncpy(anim->first, anim->name, sizeof(anim->first));
+ anim->duration = 1;
+ }
+ break;
+ case ANIM_MOVIE:
+ if (startmovie(anim))
+ return (NULL);
+ ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0); /* fake */
+ break;
#ifdef WITH_AVI
- case ANIM_AVI:
- if (startavi(anim)) {
- printf("couldn't start avi\n");
- return (NULL);
- }
- ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0);
- break;
+ case ANIM_AVI:
+ if (startavi(anim)) {
+ printf("couldn't start avi\n");
+ return (NULL);
+ }
+ ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0);
+ break;
#endif
#ifdef WITH_FFMPEG
- case ANIM_FFMPEG:
- if (startffmpeg(anim)) return (0);
- ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0);
- break;
+ case ANIM_FFMPEG:
+ if (startffmpeg(anim))
+ return (0);
+ ibuf = IMB_allocImBuf(anim->x, anim->y, 24, 0);
+ break;
#endif
- }
- return(ibuf);
+ }
+ return (ibuf);
}
struct ImBuf *IMB_anim_previewframe(struct anim *anim)
{
- struct ImBuf *ibuf = NULL;
- int position = 0;
-
- ibuf = IMB_anim_absolute(anim, 0, IMB_TC_NONE, IMB_PROXY_NONE);
- if (ibuf) {
- IMB_freeImBuf(ibuf);
- position = anim->duration / 2;
- ibuf = IMB_anim_absolute(anim, position, IMB_TC_NONE,
- IMB_PROXY_NONE);
- }
- return ibuf;
+ struct ImBuf *ibuf = NULL;
+ int position = 0;
+
+ ibuf = IMB_anim_absolute(anim, 0, IMB_TC_NONE, IMB_PROXY_NONE);
+ if (ibuf) {
+ IMB_freeImBuf(ibuf);
+ position = anim->duration / 2;
+ ibuf = IMB_anim_absolute(anim, position, IMB_TC_NONE, IMB_PROXY_NONE);
+ }
+ return ibuf;
}
-struct ImBuf *IMB_anim_absolute(struct anim *anim, int position,
+struct ImBuf *IMB_anim_absolute(struct anim *anim,
+ int position,
IMB_Timecode_Type tc,
IMB_Proxy_Size preview_size)
{
- struct ImBuf *ibuf = NULL;
- char head[256], tail[256];
- unsigned short digits;
- int pic;
- int filter_y;
- if (anim == NULL) return(NULL);
-
- filter_y = (anim->ib_flags & IB_animdeinterlace);
-
- if (preview_size == IMB_PROXY_NONE) {
- if (anim->curtype == 0) {
- ibuf = anim_getnew(anim);
- if (ibuf == NULL) {
- return(NULL);
- }
-
- IMB_freeImBuf(ibuf); /* ???? */
- ibuf = NULL;
- }
-
- if (position < 0) return(NULL);
- if (position >= anim->duration) return(NULL);
- }
- else {
- struct anim *proxy = IMB_anim_open_proxy(anim, preview_size);
-
- if (proxy) {
- position = IMB_anim_index_get_frame_index(
- anim, tc, position);
-
- return IMB_anim_absolute(
- proxy, position,
- IMB_TC_NONE, IMB_PROXY_NONE);
- }
- }
-
- switch (anim->curtype) {
- case ANIM_SEQUENCE:
- pic = an_stringdec(anim->first, head, tail, &digits);
- pic += position;
- an_stringenc(anim->name, head, tail, digits, pic);
- ibuf = IMB_loadiffname(anim->name, IB_rect, anim->colorspace);
- if (ibuf) {
- anim->curposition = position;
- }
- break;
- case ANIM_MOVIE:
- ibuf = movie_fetchibuf(anim, position);
- if (ibuf) {
- anim->curposition = position;
- IMB_convert_rgba_to_abgr(ibuf);
- }
- break;
+ struct ImBuf *ibuf = NULL;
+ char head[256], tail[256];
+ unsigned short digits;
+ int pic;
+ int filter_y;
+ if (anim == NULL)
+ return (NULL);
+
+ filter_y = (anim->ib_flags & IB_animdeinterlace);
+
+ if (preview_size == IMB_PROXY_NONE) {
+ if (anim->curtype == 0) {
+ ibuf = anim_getnew(anim);
+ if (ibuf == NULL) {
+ return (NULL);
+ }
+
+ IMB_freeImBuf(ibuf); /* ???? */
+ ibuf = NULL;
+ }
+
+ if (position < 0)
+ return (NULL);
+ if (position >= anim->duration)
+ return (NULL);
+ }
+ else {
+ struct anim *proxy = IMB_anim_open_proxy(anim, preview_size);
+
+ if (proxy) {
+ position = IMB_anim_index_get_frame_index(anim, tc, position);
+
+ return IMB_anim_absolute(proxy, position, IMB_TC_NONE, IMB_PROXY_NONE);
+ }
+ }
+
+ switch (anim->curtype) {
+ case ANIM_SEQUENCE:
+ pic = an_stringdec(anim->first, head, tail, &digits);
+ pic += position;
+ an_stringenc(anim->name, head, tail, digits, pic);
+ ibuf = IMB_loadiffname(anim->name, IB_rect, anim->colorspace);
+ if (ibuf) {
+ anim->curposition = position;
+ }
+ break;
+ case ANIM_MOVIE:
+ ibuf = movie_fetchibuf(anim, position);
+ if (ibuf) {
+ anim->curposition = position;
+ IMB_convert_rgba_to_abgr(ibuf);
+ }
+ break;
#ifdef WITH_AVI
- case ANIM_AVI:
- ibuf = avi_fetchibuf(anim, position);
- if (ibuf)
- anim->curposition = position;
- break;
+ case ANIM_AVI:
+ ibuf = avi_fetchibuf(anim, position);
+ if (ibuf)
+ anim->curposition = position;
+ break;
#endif
#ifdef WITH_FFMPEG
- case ANIM_FFMPEG:
- ibuf = ffmpeg_fetchibuf(anim, position, tc);
- if (ibuf)
- anim->curposition = position;
- filter_y = 0; /* done internally */
- break;
+ case ANIM_FFMPEG:
+ ibuf = ffmpeg_fetchibuf(anim, position, tc);
+ if (ibuf)
+ anim->curposition = position;
+ filter_y = 0; /* done internally */
+ break;
#endif
- }
-
- if (ibuf) {
- if (filter_y) IMB_filtery(ibuf);
- BLI_snprintf(ibuf->name, sizeof(ibuf->name), "%s.%04d", anim->name, anim->curposition + 1);
-
- }
- return(ibuf);
+ }
+
+ if (ibuf) {
+ if (filter_y)
+ IMB_filtery(ibuf);
+ BLI_snprintf(ibuf->name, sizeof(ibuf->name), "%s.%04d", anim->name, anim->curposition + 1);
+ }
+ return (ibuf);
}
/***/
int IMB_anim_get_duration(struct anim *anim, IMB_Timecode_Type tc)
{
- struct anim_index *idx;
- if (tc == IMB_TC_NONE) {
- return anim->duration;
- }
+ struct anim_index *idx;
+ if (tc == IMB_TC_NONE) {
+ return anim->duration;
+ }
- idx = IMB_anim_open_index(anim, tc);
- if (!idx) {
- return anim->duration;
- }
+ idx = IMB_anim_open_index(anim, tc);
+ if (!idx) {
+ return anim->duration;
+ }
- return IMB_indexer_get_duration(idx);
+ return IMB_indexer_get_duration(idx);
}
-bool IMB_anim_get_fps(struct anim *anim,
- short *frs_sec, float *frs_sec_base, bool no_av_base)
+bool IMB_anim_get_fps(struct anim *anim, short *frs_sec, float *frs_sec_base, bool no_av_base)
{
- double frs_sec_base_double;
- if (anim->frs_sec) {
- if (anim->frs_sec > SHRT_MAX) {
- /* We cannot store original rational in our short/float format,
- * we need to approximate it as best as we can... */
- *frs_sec = SHRT_MAX;
- frs_sec_base_double = anim->frs_sec_base * (double)SHRT_MAX / (double)anim->frs_sec;
- }
- else {
- *frs_sec = anim->frs_sec;
- frs_sec_base_double = anim->frs_sec_base;
- }
+ double frs_sec_base_double;
+ if (anim->frs_sec) {
+ if (anim->frs_sec > SHRT_MAX) {
+ /* We cannot store original rational in our short/float format,
+ * we need to approximate it as best as we can... */
+ *frs_sec = SHRT_MAX;
+ frs_sec_base_double = anim->frs_sec_base * (double)SHRT_MAX / (double)anim->frs_sec;
+ }
+ else {
+ *frs_sec = anim->frs_sec;
+ frs_sec_base_double = anim->frs_sec_base;
+ }
#ifdef WITH_FFMPEG
- if (no_av_base) {
- *frs_sec_base = (float)(frs_sec_base_double / AV_TIME_BASE);
- }
- else {
- *frs_sec_base = (float)frs_sec_base_double;
- }
+ if (no_av_base) {
+ *frs_sec_base = (float)(frs_sec_base_double / AV_TIME_BASE);
+ }
+ else {
+ *frs_sec_base = (float)frs_sec_base_double;
+ }
#else
- UNUSED_VARS(no_av_base);
- *frs_sec_base = (float)frs_sec_base_double;
+ UNUSED_VARS(no_av_base);
+ *frs_sec_base = (float)frs_sec_base_double;
#endif
- BLI_assert(*frs_sec > 0);
- BLI_assert(*frs_sec_base > 0.0f);
+ BLI_assert(*frs_sec > 0);
+ BLI_assert(*frs_sec_base > 0.0f);
- return true;
- }
- return false;
+ return true;
+ }
+ return false;
}
void IMB_anim_set_preseek(struct anim *anim, int preseek)
{
- anim->preseek = preseek;
+ anim->preseek = preseek;
}
int IMB_anim_get_preseek(struct anim *anim)
{
- return anim->preseek;
+ return anim->preseek;
}