2008-09-03 19:03:20 +02:00
// Copyright (c) 2008, Karl Blomster
// All rights reserved.
//
// Redistribution and use in source and binary forms, with or without
// modification, are permitted provided that the following conditions are met:
//
// * Redistributions of source code must retain the above copyright notice,
// this list of conditions and the following disclaimer.
// * Redistributions in binary form must reproduce the above copyright notice,
// this list of conditions and the following disclaimer in the documentation
// and/or other materials provided with the distribution.
// * Neither the name of the Aegisub Group nor the names of its contributors
// may be used to endorse or promote products derived from this software
// without specific prior written permission.
//
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
// AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
// IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
// ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
// LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
// CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
// SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
// INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
// CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
// ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
// POSSIBILITY OF SUCH DAMAGE.
//
// -----------------------------------------------------------------------------
//
// AEGISUB
//
// Website: http://aegisub.cellosoft.com
// Contact: mailto:zeratul@cellosoft.com
//
# ifdef WITH_FFMPEGSOURCE
///////////
// Headers
# include "video_provider_ffmpegsource.h"
# include <ffms.h>
# include "video_context.h"
2008-09-05 00:17:34 +02:00
# include "options.h"
2008-09-03 19:03:20 +02:00
///////////////
// Constructor
FFmpegSourceVideoProvider : : FFmpegSourceVideoProvider ( Aegisub : : String filename , double fps ) {
// initialize ffmpegsource
FFMS_Init ( ) ;
// clean up variables
VideoSource = NULL ;
2008-09-10 23:05:54 +02:00
DstFormat = FFMS_PIX_FMT_NONE ;
LastDstFormat = FFMS_PIX_FMT_NONE ;
2008-09-03 19:03:20 +02:00
KeyFramesLoaded = false ;
FrameNumber = - 1 ;
MessageSize = sizeof ( FFMSErrorMessage ) ;
// and here we go
LoadVideo ( filename , fps ) ;
}
///////////////
// Destructor
FFmpegSourceVideoProvider : : ~ FFmpegSourceVideoProvider ( ) {
Close ( ) ;
}
///////////////
// Open video
void FFmpegSourceVideoProvider : : LoadVideo ( Aegisub : : String filename , double fps ) {
// make sure we don't have anything messy lying around
Close ( ) ;
wxString FileNameWX ( filename . c_str ( ) , wxConvUTF8 ) ;
// generate a name for the cache file
wxString CacheName ( filename . c_str ( ) ) ;
CacheName . append ( _T ( " .ffindex " ) ) ;
2008-09-05 00:17:34 +02:00
FrameIndex * Index ;
// try to read index
Index = FFMS_ReadIndex ( CacheName . char_str ( ) , FFMSErrorMessage , MessageSize ) ;
if ( Index = = NULL ) {
2008-09-03 19:03:20 +02:00
// reading failed, create index
2008-09-05 00:17:34 +02:00
// prepare stuff for callback
IndexingProgressDialog Progress ;
Progress . IndexingCanceled = false ;
Progress . ProgressDialog = new DialogProgress ( NULL , _ ( " Indexing " ) , & Progress . IndexingCanceled , _ ( " Reading keyframes and timecodes from video " ) , 0 , 1 ) ;
Progress . ProgressDialog - > Show ( ) ;
Progress . ProgressDialog - > SetProgress ( 0 , 1 ) ;
Index = FFMS_MakeIndex ( FileNameWX . char_str ( ) , 0 , " " , FFmpegSourceVideoProvider : : UpdateIndexingProgress , & Progress , FFMSErrorMessage , MessageSize ) ;
if ( Index = = NULL ) {
2008-09-03 19:03:20 +02:00
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
2008-09-05 00:17:34 +02:00
Progress . ProgressDialog - > Destroy ( ) ;
2008-09-03 19:03:20 +02:00
// write it to disk
if ( FFMS_WriteIndex ( CacheName . char_str ( ) , Index , FFMSErrorMessage , MessageSize ) ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
}
2008-09-03 23:22:33 +02:00
// set thread count
int Threads = Options . AsInt ( _T ( " FFmpegSource decoding threads " ) ) ;
2008-09-03 19:03:20 +02:00
if ( Threads < 1 )
throw _T ( " FFmpegSource video provider: invalid decoding thread count " ) ;
2008-09-03 23:22:33 +02:00
// set seekmode
// TODO: give this its own option?
int SeekMode ;
if ( Options . AsBool ( _T ( " FFmpeg allow unsafe seeking " ) ) )
SeekMode = 2 ;
else
SeekMode = 1 ;
2008-09-03 19:03:20 +02:00
// finally create the actual video source
VideoSource = FFMS_CreateVideoSource ( FileNameWX . char_str ( ) , - 1 , Index , " " , Threads , SeekMode , FFMSErrorMessage , MessageSize ) ;
if ( VideoSource = = NULL ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
// load video properties
VideoInfo = FFMS_GetVideoProperties ( VideoSource ) ;
// get frame info data
2008-09-03 22:27:50 +02:00
FrameInfoVector * FrameData = FFMS_GetVSTrackIndex ( VideoSource , FFMSErrorMessage , MessageSize ) ;
2008-09-03 19:03:20 +02:00
if ( FrameData = = NULL ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
2008-09-03 22:27:50 +02:00
const TrackTimeBase * TimeBase = FFMS_GetTimeBase ( FrameData , FFMSErrorMessage , MessageSize ) ;
if ( TimeBase = = NULL ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
const FrameInfo * CurFrameData ;
2008-09-03 19:03:20 +02:00
2008-09-03 22:27:50 +02:00
// build list of keyframes and timecodes
2008-09-03 19:03:20 +02:00
for ( int CurFrameNum = 0 ; CurFrameNum < VideoInfo - > NumFrames ; CurFrameNum + + ) {
CurFrameData = FFMS_GetFrameInfo ( FrameData , CurFrameNum , FFMSErrorMessage , MessageSize ) ;
if ( CurFrameData = = NULL ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
2008-09-03 22:27:50 +02:00
// keyframe?
if ( CurFrameData - > KeyFrame )
2008-09-03 19:03:20 +02:00
KeyFramesList . Add ( CurFrameNum ) ;
2008-09-03 22:27:50 +02:00
// calculate timestamp and add to timecodes vector
int64_t Timestamp = ( int64_t ) ( ( CurFrameData - > DTS * TimeBase - > Num ) / ( double ) TimeBase - > Den ) ;
2008-09-06 04:54:22 +02:00
TimecodesVector . push_back ( Timestamp ) ;
2008-09-03 19:03:20 +02:00
}
KeyFramesLoaded = true ;
2008-09-03 22:27:50 +02:00
// override already loaded timecodes?
Timecodes . SetVFR ( TimecodesVector ) ;
int OverrideTC = wxYES ;
if ( VFR_Output . IsLoaded ( ) ) {
OverrideTC = wxMessageBox ( _ ( " You already have timecodes loaded. Would you like to replace them with timecodes from the video file? " ) , _ ( " Replace timecodes? " ) , wxYES_NO | wxICON_QUESTION ) ;
if ( OverrideTC = = wxYES ) {
VFR_Input . SetVFR ( TimecodesVector ) ;
VFR_Output . SetVFR ( TimecodesVector ) ;
}
2008-09-03 23:03:18 +02:00
} else { // no timecodes loaded, go ahead and apply
VFR_Input . SetVFR ( TimecodesVector ) ;
VFR_Output . SetVFR ( TimecodesVector ) ;
2008-09-03 22:27:50 +02:00
}
2008-09-03 19:03:20 +02:00
// we don't need this anymore
FFMS_DestroyFrameIndex ( Index ) ;
FrameNumber = 0 ;
}
///////////////
// Close video
void FFmpegSourceVideoProvider : : Close ( ) {
if ( VideoSource )
FFMS_DestroyVideoSource ( VideoSource ) ;
VideoSource = NULL ;
2008-09-10 23:05:54 +02:00
DstFormat = FFMS_PIX_FMT_NONE ;
LastDstFormat = FFMS_PIX_FMT_NONE ;
2008-09-03 19:03:20 +02:00
KeyFramesLoaded = false ;
KeyFramesList . clear ( ) ;
2008-09-03 22:27:50 +02:00
TimecodesVector . clear ( ) ;
2008-09-03 19:03:20 +02:00
FrameNumber = - 1 ;
}
2008-09-05 00:17:34 +02:00
///////////////
// Update indexing progress
int __stdcall FFmpegSourceVideoProvider : : UpdateIndexingProgress ( int State , int64_t Current , int64_t Total , void * Private ) {
IndexingProgressDialog * Progress = ( IndexingProgressDialog * ) Private ;
2008-09-07 21:15:12 +02:00
if ( Progress - > IndexingCanceled )
2008-09-05 00:17:34 +02:00
return 1 ;
2008-09-07 21:15:12 +02:00
// noone cares about a little bit of a rounding error here anyway
Progress - > ProgressDialog - > SetProgress ( ( 1000 * Current ) / Total , 1000 ) ;
return 0 ;
2008-09-05 00:17:34 +02:00
}
2008-09-03 19:03:20 +02:00
///////////////
// Get frame
2008-09-03 23:03:18 +02:00
const AegiVideoFrame FFmpegSourceVideoProvider : : GetFrame ( int _n , int FormatType ) {
// don't try to seek to insane places
int n = _n ;
if ( n < 0 )
n = 0 ;
if ( n > = GetFrameCount ( ) )
n = GetFrameCount ( ) - 1 ;
// set position
FrameNumber = n ;
2008-09-10 23:05:54 +02:00
// these are for convenience
int w = VideoInfo - > Width ;
int h = VideoInfo - > Height ;
// this is what we'll return eventually
AegiVideoFrame & DstFrame = CurFrame ;
// choose output format
if ( FormatType & FORMAT_RGB32 ) {
DstFormat = FFMS_PIX_FMT_RGB32 ; // FIXME: should be RGB32
DstFrame . format = FORMAT_RGB32 ;
} else if ( FormatType & FORMAT_RGB24 ) {
DstFormat = FFMS_PIX_FMT_RGB24 ;
DstFrame . format = FORMAT_RGB24 ;
} else if ( FormatType & FORMAT_YV12 ) {
DstFormat = FFMS_PIX_FMT_YUV420P ; // may or may not work
DstFrame . format = FORMAT_YV12 ;
} else if ( FormatType & FORMAT_YUY2 ) {
DstFormat = FFMS_PIX_FMT_YUYV422 ;
DstFrame . format = FORMAT_YUY2 ;
} else
throw _T ( " FFmpegSource video provider: upstream provider requested unknown or unsupported pixel format " ) ;
// requested format was changed since last time we were called, (re)set output format
if ( LastDstFormat ! = DstFormat ) {
if ( FFMS_SetOutputFormat ( VideoSource , DstFormat , w , h ) )
throw _T ( " FFmpegSource video provider: failed to set desired output format " ) ;
LastDstFormat = DstFormat ;
}
2008-09-03 23:03:18 +02:00
2008-09-10 23:05:54 +02:00
// decode frame
2008-09-03 19:03:20 +02:00
const AVFrameLite * SrcFrame = FFMS_GetFrame ( VideoSource , n , FFMSErrorMessage , MessageSize ) ;
if ( SrcFrame = = NULL ) {
ErrorMsg . Printf ( _T ( " FFmpegSource video provider: %s " ) , FFMSErrorMessage ) ;
throw ErrorMsg ;
}
// set some properties
DstFrame . w = w ;
DstFrame . h = h ;
DstFrame . flipped = false ;
2008-09-10 23:05:54 +02:00
if ( DstFrame . format = = FORMAT_RGB32 | | DstFrame . format = = FORMAT_RGB24 )
2008-09-03 22:27:50 +02:00
DstFrame . invertChannels = true ;
2008-09-10 23:05:54 +02:00
else
2008-09-03 22:27:50 +02:00
DstFrame . invertChannels = false ;
2008-09-03 19:03:20 +02:00
2008-09-10 23:05:54 +02:00
// allocate destination
for ( int i = 0 ; i < 4 ; i + + )
DstFrame . pitch [ i ] = SrcFrame - > Linesize [ i ] ;
2008-09-03 19:03:20 +02:00
DstFrame . Allocate ( ) ;
2008-09-10 23:05:54 +02:00
// copy data to destination, skipping planes with no data in them
for ( int j = 0 ; j < 4 ; j + + ) {
if ( SrcFrame - > Linesize [ j ] > 0 )
memcpy ( DstFrame . data [ j ] , SrcFrame - > Data [ j ] , DstFrame . pitch [ j ] * DstFrame . h ) ;
}
2008-09-03 19:03:20 +02:00
return DstFrame ;
}
///////////////
// Utility functions
int FFmpegSourceVideoProvider : : GetWidth ( ) {
return VideoInfo - > Width ;
}
int FFmpegSourceVideoProvider : : GetHeight ( ) {
return VideoInfo - > Height ;
}
int FFmpegSourceVideoProvider : : GetFrameCount ( ) {
return VideoInfo - > NumFrames ;
}
int FFmpegSourceVideoProvider : : GetPosition ( ) {
return FrameNumber ;
}
double FFmpegSourceVideoProvider : : GetFPS ( ) {
return double ( VideoInfo - > FPSNumerator ) / double ( VideoInfo - > FPSDenominator ) ;
}
2008-09-05 00:17:34 +02:00
# endif /* WITH_FFMPEGSOURCE */