2001-12-08 12:21:23 +00:00
# define VCODEC_COPY 0
2001-12-06 23:59:46 +00:00
# define VCODEC_FRAMENO 1
2002-04-11 02:52:03 +00:00
// real codecs:
2001-12-06 23:59:46 +00:00
# define VCODEC_DIVX4 2
2001-12-15 17:12:20 +00:00
# define VCODEC_LIBAVCODEC 4
2002-01-31 00:38:53 +00:00
# define VCODEC_RAWRGB 6
2002-02-07 20:22:03 +00:00
# define VCODEC_VFW 7
2002-04-12 21:50:38 +00:00
# define VCODEC_LIBDV 8
2001-12-06 23:59:46 +00:00
2001-12-08 13:30:06 +00:00
# define ACODEC_COPY 0
2001-10-31 00:25:28 +00:00
# define ACODEC_PCM 1
# define ACODEC_VBRMP3 2
2002-01-27 00:43:57 +00:00
# define ACODEC_NULL 3
2001-10-29 01:11:18 +00:00
# include <stdio.h>
# include <stdlib.h>
# include <string.h>
# include <signal.h>
2002-04-12 10:40:38 +00:00
# include <sys/time.h>
2001-10-29 01:11:18 +00:00
2001-10-31 16:08:01 +00:00
# include "config.h"
2001-10-29 01:11:18 +00:00
# include "mp_msg.h"
2001-11-18 19:12:25 +00:00
# include "version.h"
2001-10-29 01:11:18 +00:00
# include "help_mp.h"
2001-11-18 19:12:25 +00:00
static char * banner_text =
" \n \n "
2002-02-18 20:17:36 +00:00
" MEncoder " VERSION " (C) 2000-2002 Arpad Gereoffy (see DOCS!) \n "
2001-11-18 19:12:25 +00:00
" \n " ;
2001-12-04 21:35:18 +00:00
# include "cpudetect.h"
2001-10-29 01:11:18 +00:00
# include "codec-cfg.h"
2002-01-24 23:02:59 +00:00
# include "cfgparser.h"
2001-10-29 01:11:18 +00:00
# include "stream.h"
# include "demuxer.h"
# include "stheader.h"
2002-01-14 23:38:49 +00:00
# include "playtree.h"
2001-10-29 01:11:18 +00:00
# include "aviwrite.h"
# include "libvo/video_out.h"
2002-04-13 19:14:34 +00:00
# include "libmpcodecs/dec_audio.h"
# include "libmpcodecs/dec_video.h"
2001-10-30 21:55:28 +00:00
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-10-31 16:08:01 +00:00
# include <lame/lame.h>
2001-12-06 22:27:09 +00:00
# endif
2001-10-31 00:25:28 +00:00
2001-12-16 11:37:23 +00:00
# ifdef HAVE_LIBCSS
# include "libmpdemux/dvdauth.h"
# endif
2001-12-01 01:29:25 +00:00
# include <inttypes.h>
2001-12-08 13:30:06 +00:00
# include "fastmemcpy.h"
2002-01-27 18:39:53 +00:00
/**************************************************************************
Video accelerated architecture
* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2002-04-11 02:52:03 +00:00
vo_vaa_t vo_vaa ; // FIXME! remove me!
2002-04-07 02:12:15 +00:00
int vo_doublebuffering = 0 ;
int vo_directrendering = 0 ;
int vo_config_count = 0 ;
2002-01-27 18:39:53 +00:00
2001-10-29 01:11:18 +00:00
//--------------------------
// cache2:
2001-11-02 01:25:13 +00:00
static int stream_cache_size = 0 ;
2001-10-29 01:11:18 +00:00
# ifdef USE_STREAM_CACHE
extern int cache_fill_status ;
# else
# define cache_fill_status 0
# endif
2001-11-02 01:25:13 +00:00
int vcd_track = 0 ;
int audio_id = - 1 ;
int video_id = - 1 ;
int dvdsub_id = - 1 ;
2002-03-29 03:17:57 +00:00
int vobsub_id = - 1 ;
2002-03-31 22:01:54 +00:00
char * audio_lang = NULL ;
char * dvdsub_lang = NULL ;
2002-03-29 03:17:57 +00:00
static char * spudec_ifo = NULL ;
2001-11-02 01:25:13 +00:00
2002-01-26 16:32:06 +00:00
static int has_audio = 1 ;
2001-10-29 01:11:18 +00:00
char * audio_codec = NULL ; // override audio codec
char * video_codec = NULL ; // override video codec
int audio_family = - 1 ; // override audio codec family
int video_family = - 1 ; // override video codec family
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-11-03 23:01:17 +00:00
int out_audio_codec = ACODEC_VBRMP3 ;
2001-12-06 22:27:09 +00:00
# else
int out_audio_codec = ACODEC_PCM ;
# endif
2002-02-10 00:07:34 +00:00
int out_video_codec =
# ifdef HAVE_DIVX4ENCORE
VCODEC_DIVX4 ;
# else
# ifdef USE_LIBAVCODEC
VCODEC_LIBAVCODEC ;
# else
2002-04-12 00:15:03 +00:00
VCODEC_COPY ;
2002-02-10 00:07:34 +00:00
# endif
# endif
2001-11-03 23:01:17 +00:00
2001-10-29 01:11:18 +00:00
// audio stream skip/resync functions requires only for seeking.
// (they should be implemented in the audio codec layer)
//void skip_audio_frame(sh_audio_t *sh_audio){}
//void resync_audio_stream(sh_audio_t *sh_audio){}
2001-11-02 01:25:13 +00:00
int verbose = 0 ; // must be global!
2001-10-29 01:11:18 +00:00
double video_time_usage = 0 ;
double vout_time_usage = 0 ;
2002-02-24 11:23:48 +00:00
double max_video_time_usage = 0 ;
double max_vout_time_usage = 0 ;
2002-02-24 15:54:13 +00:00
double cur_video_time_usage = 0 ;
double cur_vout_time_usage = 0 ;
2001-10-29 01:11:18 +00:00
static double audio_time_usage = 0 ;
static int total_time_usage_start = 0 ;
2002-02-24 17:10:06 +00:00
int benchmark = 0 ;
2001-10-29 01:11:18 +00:00
2001-11-01 02:31:23 +00:00
// A-V sync:
int delay_corrected = 1 ;
static float default_max_pts_correction = - 1 ; //0.01f;
static float max_pts_correction = 0 ; //default_max_pts_correction;
static float c_total = 0 ;
2001-11-01 21:47:28 +00:00
float force_fps = 0 ;
float force_ofps = 0 ; // set to 24 for inverse telecine
2002-04-15 02:33:05 +00:00
static int skip_limit = - 1 ;
2001-10-29 01:11:18 +00:00
2001-11-02 01:25:13 +00:00
int force_srate = 0 ;
2001-11-02 03:22:33 +00:00
char * out_filename = " test.avi " ;
char * mp3_filename = NULL ;
char * ac3_filename = NULL ;
2001-12-22 14:32:08 +00:00
char * force_fourcc = NULL ;
2002-04-11 02:52:03 +00:00
int pass = 0 ;
char * passtmpfile = " divx2pass.log " ;
2001-12-08 01:43:02 +00:00
int pass_working = 0 ;
2001-11-03 00:44:02 +00:00
static int play_n_frames = - 1 ;
2002-04-17 21:51:18 +00:00
# include "libvo/font_load.h"
# include "libvo/sub.h"
// sub:
char * font_name = NULL ;
float font_factor = 0.75 ;
char * sub_name = NULL ;
float sub_delay = 0 ;
float sub_fps = 0 ;
int sub_auto = 0 ;
# ifdef USE_SUB
static subtitle * subtitles = NULL ;
float sub_last_pts = - 303 ;
# endif
2001-11-03 23:01:17 +00:00
//char *out_audio_codec=NULL; // override audio codec
//char *out_video_codec=NULL; // override video codec
2001-11-02 03:22:33 +00:00
2001-10-31 16:08:01 +00:00
//#include "libmpeg2/mpeg2.h"
//#include "libmpeg2/mpeg2_internal.h"
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-11-02 03:22:33 +00:00
int lame_param_quality = 0 ; // best
int lame_param_vbr = vbr_default ;
int lame_param_mode = - 1 ; // unset
int lame_param_padding = - 1 ; // unset
int lame_param_br = - 1 ; // unset
int lame_param_ratio = - 1 ; // unset
2001-12-06 22:27:09 +00:00
# endif
2001-11-02 03:22:33 +00:00
2001-12-01 01:29:25 +00:00
static int vo_w = 0 , vo_h = 0 ;
2002-03-23 13:13:12 +00:00
static int input_pitch , input_bpp ;
2001-11-02 01:25:13 +00:00
//-------------------------- config stuff:
2002-01-14 23:38:49 +00:00
m_config_t * mconfig ;
2001-11-02 01:25:13 +00:00
static int cfg_inc_verbose ( struct config * conf ) { + + verbose ; return 0 ; }
static int cfg_include ( struct config * conf , char * filename ) {
2002-01-14 23:38:49 +00:00
return m_config_parse_config_file ( mconfig , filename ) ;
2001-11-02 01:25:13 +00:00
}
2002-02-10 00:07:34 +00:00
static char * seek_to_sec = NULL ;
static off_t seek_to_byte = 0 ;
2002-01-15 00:20:50 +00:00
static int parse_end_at ( struct config * conf , const char * param ) ;
2002-01-31 00:38:53 +00:00
static uint8_t * flip_upside_down ( uint8_t * dst , const uint8_t * src , int width , int height ) ;
2002-01-15 00:20:50 +00:00
2001-11-02 01:25:13 +00:00
# include "get_path.c"
# include "cfg-mplayer-def.h"
# include "cfg-mencoder.h"
2002-01-11 16:06:45 +00:00
# ifdef USE_DVDREAD
# include "spudec.h"
# endif
2002-03-29 03:17:57 +00:00
# include "vobsub.h"
2002-01-11 16:06:45 +00:00
2002-02-17 13:50:26 +00:00
/* FIXME */
void mencoder_exit ( int level , char * how )
{
if ( how )
printf ( " Exiting... (%s) \n " , how ) ;
else
printf ( " Exiting... \n " ) ;
exit ( level ) ;
}
2002-02-02 23:39:53 +00:00
void parse_cfgfiles ( m_config_t * conf )
{
char * conffile ;
if ( ( conffile = get_path ( " mencoder " ) ) = = NULL ) {
mp_msg ( MSGT_CPLAYER , MSGL_ERR , MSGTR_GetpathProblem ) ;
} else {
if ( m_config_parse_config_file ( conf , conffile ) < 0 )
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , " configfile error " ) ;
2002-02-02 23:39:53 +00:00
free ( conffile ) ;
}
}
2002-04-12 10:40:38 +00:00
2001-10-31 16:08:01 +00:00
//---------------------------------------------------------------------------
int dec_audio ( sh_audio_t * sh_audio , unsigned char * buffer , int total ) {
int size = 0 ;
int eof = 0 ;
while ( size < total & & ! eof ) {
int len = total - size ;
if ( len > MAX_OUTBURST ) len = MAX_OUTBURST ;
if ( len > sh_audio - > a_buffer_size ) len = sh_audio - > a_buffer_size ;
if ( len > sh_audio - > a_buffer_len ) {
int ret = decode_audio ( sh_audio ,
& sh_audio - > a_buffer [ sh_audio - > a_buffer_len ] ,
len - sh_audio - > a_buffer_len ,
sh_audio - > a_buffer_size - sh_audio - > a_buffer_len ) ;
if ( ret > 0 ) sh_audio - > a_buffer_len + = ret ; else eof = 1 ;
}
if ( len > sh_audio - > a_buffer_len ) len = sh_audio - > a_buffer_len ;
memcpy ( buffer + size , sh_audio - > a_buffer , len ) ;
sh_audio - > a_buffer_len - = len ; size + = len ;
if ( sh_audio - > a_buffer_len > 0 )
memcpy ( sh_audio - > a_buffer , & sh_audio - > a_buffer [ len ] , sh_audio - > a_buffer_len ) ;
}
return size ;
}
2002-02-20 17:02:31 +00:00
extern void me_register_options ( m_config_t * cfg ) ;
2001-10-31 16:08:01 +00:00
//---------------------------------------------------------------------------
2001-10-29 01:11:18 +00:00
static int eof = 0 ;
2001-12-04 20:35:31 +00:00
static int interrupted = 0 ;
2001-10-29 01:11:18 +00:00
2002-01-15 00:20:50 +00:00
enum end_at_type_t { END_AT_NONE , END_AT_TIME , END_AT_SIZE } ;
static enum end_at_type_t end_at_type = END_AT_NONE ;
static int end_at ;
2001-10-29 01:11:18 +00:00
static void exit_sighandler ( int x ) {
eof = 1 ;
2001-12-04 20:35:31 +00:00
interrupted = 1 ;
2001-10-29 01:11:18 +00:00
}
2002-04-11 02:52:03 +00:00
aviwrite_t * muxer = NULL ;
FILE * muxer_f = NULL ;
// callback for ve_*.c:
void mencoder_write_chunk ( aviwrite_stream_t * s , int len , unsigned int flags ) {
aviwrite_write_chunk ( muxer , s , muxer_f , len , flags ) ;
}
2001-11-02 01:25:13 +00:00
int main ( int argc , char * argv [ ] , char * envp [ ] ) {
2001-10-29 01:11:18 +00:00
stream_t * stream = NULL ;
demuxer_t * demuxer = NULL ;
2002-01-26 22:30:02 +00:00
stream_t * stream2 = NULL ;
demuxer_t * demuxer2 = NULL ;
2001-10-29 01:11:18 +00:00
demux_stream_t * d_audio = NULL ;
demux_stream_t * d_video = NULL ;
demux_stream_t * d_dvdsub = NULL ;
sh_audio_t * sh_audio = NULL ;
sh_video_t * sh_video = NULL ;
int file_format = DEMUXER_TYPE_UNKNOWN ;
2002-03-16 22:51:07 +00:00
int i ;
2001-10-29 01:11:18 +00:00
2002-04-12 10:40:38 +00:00
uint32_t ptimer_start ;
uint32_t audiorate = 0 ;
uint32_t videorate = 0 ;
uint32_t audiosamples = 1 ;
uint32_t videosamples = 1 ;
uint32_t skippedframes = 0 ;
uint32_t duplicatedframes = 0 ;
2001-10-29 01:11:18 +00:00
aviwrite_stream_t * mux_a = NULL ;
aviwrite_stream_t * mux_v = NULL ;
2002-04-12 10:40:38 +00:00
off_t muxer_f_size = 0 ;
2001-10-29 01:11:18 +00:00
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-10-31 16:08:01 +00:00
lame_global_flags * lame ;
2001-12-06 22:27:09 +00:00
# endif
2001-10-31 00:25:28 +00:00
2001-11-03 21:00:04 +00:00
float audio_preload = 0.5 ;
2001-10-30 23:17:35 +00:00
2001-11-01 21:47:28 +00:00
double v_pts_corr = 0 ;
double v_timer_corr = 0 ;
2001-11-01 02:31:23 +00:00
2002-01-14 23:38:49 +00:00
play_tree_t * playtree ;
play_tree_iter_t * playtree_iter ;
2001-11-02 01:25:13 +00:00
char * filename = NULL ;
2002-01-26 22:30:02 +00:00
char * frameno_filename = " frameno.avi " ;
2001-11-02 01:25:13 +00:00
2001-12-06 23:59:46 +00:00
int decoded_frameno = 0 ;
2002-01-26 22:30:02 +00:00
int next_frameno = - 1 ;
2001-12-06 23:59:46 +00:00
2002-01-27 18:29:33 +00:00
unsigned int timer_start ;
2002-03-20 15:38:13 +00:00
mp_msg_init ( ) ;
mp_msg_set_level ( MSGL_STATUS ) ;
2001-11-18 19:12:25 +00:00
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " %s " , banner_text ) ;
2001-11-02 02:08:00 +00:00
// check codec.conf
if ( ! parse_codec_cfg ( get_path ( " codecs.conf " ) ) ) {
2001-12-25 21:59:07 +00:00
if ( ! parse_codec_cfg ( CONFDIR " /codecs.conf " ) ) {
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_HINT , MSGTR_CopyCodecsConf ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-11-02 02:08:00 +00:00
}
}
2001-10-29 01:11:18 +00:00
2001-12-04 21:35:18 +00:00
/* Test for cpu capabilities (and corresponding OS support) for optimizing */
# ifdef ARCH_X86
GetCpuCaps ( & gCpuCaps ) ;
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " CPUflags: Type: %d MMX: %d MMX2: %d 3DNow: %d 3DNow2: %d SSE: %d SSE2: %d \n " ,
gCpuCaps . cpuType , gCpuCaps . hasMMX , gCpuCaps . hasMMX2 ,
gCpuCaps . has3DNow , gCpuCaps . has3DNowExt ,
gCpuCaps . hasSSE , gCpuCaps . hasSSE2 ) ;
# endif
2002-04-17 21:51:18 +00:00
// check font
# ifdef USE_OSD
if ( font_name ) {
vo_font = read_font_desc ( font_name , font_factor , verbose > 1 ) ;
if ( ! vo_font ) mp_msg ( MSGT_CPLAYER , MSGL_ERR , MSGTR_CantLoadFont , font_name ) ;
} else {
// try default:
vo_font = read_font_desc ( get_path ( " font/font.desc " ) , font_factor , verbose > 1 ) ;
if ( ! vo_font )
vo_font = read_font_desc ( DATADIR " /font/font.desc " , font_factor , verbose > 1 ) ;
}
# endif
vo_init_osd ( ) ;
2002-01-26 22:30:02 +00:00
// FIXME: get rid of -dvd and other tricky options and config/playtree
stream2 = open_stream ( frameno_filename , 0 , & i ) ;
if ( stream2 ) {
demuxer2 = demux_open ( stream2 , DEMUXER_TYPE_AVI , - 1 , - 1 , - 2 ) ;
if ( demuxer2 ) printf ( " Using pass3 control file: %s \n " , frameno_filename ) ;
}
2002-01-14 23:38:49 +00:00
playtree = play_tree_new ( ) ;
mconfig = m_config_new ( playtree ) ;
m_config_register_options ( mconfig , mencoder_opts ) ;
2002-02-20 17:02:31 +00:00
me_register_options ( mconfig ) ;
2002-02-02 23:39:53 +00:00
parse_cfgfiles ( mconfig ) ;
2002-01-14 23:38:49 +00:00
2002-02-17 13:50:26 +00:00
if ( m_config_parse_command_line ( mconfig , argc , argv , envp ) < 0 ) mencoder_exit ( 1 , " error parsing cmdline " ) ;
2002-01-14 23:38:49 +00:00
playtree = play_tree_cleanup ( playtree ) ;
if ( playtree ) {
playtree_iter = play_tree_iter_new ( playtree , mconfig ) ;
if ( playtree_iter ) {
if ( play_tree_iter_step ( playtree_iter , 0 , 0 ) ! = PLAY_TREE_ITER_ENTRY ) {
play_tree_iter_free ( playtree_iter ) ;
playtree_iter = NULL ;
}
filename = play_tree_iter_get_file ( playtree_iter , 1 ) ;
}
}
if ( ! filename & & ! vcd_track & & ! dvd_title & & ! tv_param_on ) {
2001-11-02 01:25:13 +00:00
printf ( " \n Missing filename! \n \n " ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-11-02 01:25:13 +00:00
}
2002-03-20 15:38:13 +00:00
mp_msg_set_level ( verbose + MSGL_STATUS ) ;
2001-10-31 22:59:56 +00:00
2001-11-02 01:25:13 +00:00
stream = open_stream ( filename , vcd_track , & file_format ) ;
2001-10-29 01:11:18 +00:00
if ( ! stream ) {
printf ( " Cannot open file/device \n " ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-10-29 01:11:18 +00:00
}
printf ( " success: format: %d data: 0x%X - 0x%X \n " , file_format , ( int ) ( stream - > start_pos ) , ( int ) ( stream - > end_pos ) ) ;
2002-03-31 22:01:54 +00:00
# ifdef USE_DVDREAD
if ( stream - > type = = STREAMTYPE_DVD ) {
if ( audio_lang & & audio_id = = - 1 ) audio_id = dvd_aid_from_lang ( stream , audio_lang ) ;
if ( dvdsub_lang & & dvdsub_id = = - 1 ) dvdsub_id = dvd_sid_from_lang ( stream , dvdsub_lang ) ;
}
# endif
2002-04-15 00:36:21 +00:00
stream - > start_pos + = seek_to_byte ;
2001-10-29 01:11:18 +00:00
2002-01-04 14:03:02 +00:00
# ifdef HAVE_LIBCSS
// current_module="libcss";
if ( dvdimportkey ) {
if ( dvd_import_key ( dvdimportkey ) ) {
mp_msg ( MSGT_CPLAYER , MSGL_FATAL , MSGTR_ErrorDVDkey ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2002-01-04 14:03:02 +00:00
}
mp_msg ( MSGT_CPLAYER , MSGL_INFO , MSGTR_CmdlineDVDkey ) ;
}
if ( dvd_auth_device ) {
// if (dvd_auth(dvd_auth_device,f)) {
if ( dvd_auth ( dvd_auth_device , filename ) ) {
mp_msg ( MSGT_CPLAYER , MSGL_FATAL , " Error in DVD auth... \n " ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2002-01-04 14:03:02 +00:00
}
mp_msg ( MSGT_CPLAYER , MSGL_INFO , MSGTR_DVDauthOk ) ;
}
# endif
2002-04-15 00:36:21 +00:00
if ( stream_cache_size ) stream_enable_cache ( stream , stream_cache_size * 1024 , 0 , 0 ) ;
2002-01-26 22:30:02 +00:00
if ( ! has_audio | | demuxer2 ) audio_id = - 2 ; /* do NOT read audio packets... */
2002-01-26 16:32:06 +00:00
2001-11-13 21:43:29 +00:00
//demuxer=demux_open(stream,file_format,video_id,audio_id,dvdsub_id);
demuxer = demux_open ( stream , file_format , audio_id , video_id , dvdsub_id ) ;
2001-10-29 01:11:18 +00:00
if ( ! demuxer ) {
printf ( " Cannot open demuxer \n " ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-10-29 01:11:18 +00:00
}
2002-01-26 22:30:02 +00:00
d_audio = demuxer2 ? demuxer2 - > audio : demuxer - > audio ;
2001-10-29 01:11:18 +00:00
d_video = demuxer - > video ;
d_dvdsub = demuxer - > sub ;
sh_audio = d_audio - > sh ;
sh_video = d_video - > sh ;
if ( ! video_read_properties ( sh_video ) ) {
printf ( " Couldn't read video properties \n " ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-10-29 01:11:18 +00:00
}
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_INFO , " [V] filefmt:%d fourcc:0x%X size:%dx%d fps:%5.2f ftime:=%6.4f \n " ,
2001-10-29 01:11:18 +00:00
demuxer - > file_format , sh_video - > format , sh_video - > disp_w , sh_video - > disp_h ,
sh_video - > fps , sh_video - > frametime
) ;
2002-03-18 01:51:47 +00:00
2002-02-10 00:07:34 +00:00
if ( sh_audio & & ( out_audio_codec | | seek_to_sec | | ! sh_audio - > wf ) ) {
2001-10-30 23:17:35 +00:00
// Go through the codec.conf and find the best codec...
sh_audio - > codec = NULL ;
2001-11-02 02:08:00 +00:00
if ( audio_family ! = - 1 ) mp_msg ( MSGT_MENCODER , MSGL_INFO , MSGTR_TryForceAudioFmt , audio_family ) ;
2001-10-30 23:17:35 +00:00
while ( 1 ) {
sh_audio - > codec = find_codec ( sh_audio - > format , NULL , sh_audio - > codec , 1 ) ;
if ( ! sh_audio - > codec ) {
if ( audio_family ! = - 1 ) {
sh_audio - > codec = NULL ; /* re-search */
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_ERR , MSGTR_CantFindAfmtFallback ) ;
2001-10-30 23:17:35 +00:00
audio_family = - 1 ;
continue ;
}
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_ERR , MSGTR_CantFindAudioCodec , sh_audio - > format ) ;
mp_msg ( MSGT_MENCODER , MSGL_HINT , MSGTR_TryUpgradeCodecsConfOrRTFM , get_path ( " codecs.conf " ) ) ;
2001-10-30 23:17:35 +00:00
sh_audio = d_audio - > sh = NULL ;
break ;
}
if ( audio_codec & & strcmp ( sh_audio - > codec - > name , audio_codec ) ) continue ;
else if ( audio_family ! = - 1 & & sh_audio - > codec - > driver ! = audio_family ) continue ;
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_INFO , " %s audio codec: [%s] drv:%d (%s) \n " , audio_codec ? " Forcing " : " Detected " , sh_audio - > codec - > name , sh_audio - > codec - > driver , sh_audio - > codec - > info ) ;
2001-10-30 23:17:35 +00:00
break ;
}
}
2002-02-10 00:07:34 +00:00
if ( sh_audio & & ( out_audio_codec | | seek_to_sec | | ! sh_audio - > wf ) ) {
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_V , " Initializing audio codec... \n " ) ;
2001-10-30 23:17:35 +00:00
if ( ! init_audio ( sh_audio ) ) {
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_ERR , MSGTR_CouldntInitAudioCodec ) ;
2001-10-30 23:17:35 +00:00
sh_audio = d_audio - > sh = NULL ;
} else {
2001-11-02 02:08:00 +00:00
mp_msg ( MSGT_MENCODER , MSGL_INFO , " AUDIO: srate=%d chans=%d bps=%d sfmt=0x%X ratio: %d->%d \n " , sh_audio - > samplerate , sh_audio - > channels , sh_audio - > samplesize ,
2001-10-30 23:17:35 +00:00
sh_audio - > sample_format , sh_audio - > i_bps , sh_audio - > o_bps ) ;
}
}
2001-10-29 01:11:18 +00:00
// set up video encoder:
2002-03-29 03:17:57 +00:00
if ( spudec_ifo ) {
unsigned int palette [ 16 ] , width , height ;
if ( vobsub_parse_ifo ( spudec_ifo , palette , & width , & height , 1 ) > = 0 )
vo_spudec = spudec_new_scaled ( palette , sh_video - > disp_w , sh_video - > disp_h ) ;
}
2002-01-11 16:06:45 +00:00
# ifdef USE_DVDREAD
2002-03-29 15:12:41 +00:00
if ( vo_spudec = = NULL ) {
2002-02-06 22:55:47 +00:00
vo_spudec = spudec_new_scaled ( stream - > type = = STREAMTYPE_DVD ? ( ( dvd_priv_t * ) ( stream - > priv ) ) - > cur_pgc - > palette : NULL ,
2002-01-11 16:06:45 +00:00
sh_video - > disp_w , sh_video - > disp_h ) ;
2002-03-29 03:17:57 +00:00
}
2002-01-11 16:06:45 +00:00
# endif
2002-04-17 21:51:18 +00:00
# ifdef USE_SUB
// after reading video params we should load subtitles because
// we know fps so now we can adjust subtitles time to ~6 seconds AST
// check .sub
// current_module="read_subtitles_file";
if ( sub_name ) {
subtitles = sub_read_file ( sub_name , sh_video - > fps ) ;
if ( ! subtitles ) mp_msg ( MSGT_CPLAYER , MSGL_ERR , MSGTR_CantLoadSub , sub_name ) ;
} else
if ( sub_auto ) { // auto load sub file ...
subtitles = sub_read_file ( filename ? sub_filename ( get_path ( " sub/ " ) , filename )
: " default.sub " , sh_video - > fps ) ;
}
# endif
2001-10-29 01:11:18 +00:00
// set up output file:
2001-11-02 03:22:33 +00:00
muxer_f = fopen ( out_filename , " wb " ) ;
2001-11-13 23:43:33 +00:00
if ( ! muxer_f ) {
printf ( " Cannot open output file '%s' \n " , out_filename ) ;
2002-02-17 13:50:26 +00:00
mencoder_exit ( 1 , NULL ) ;
2001-11-13 23:43:33 +00:00
}
2001-10-29 01:11:18 +00:00
muxer = aviwrite_new_muxer ( ) ;
2001-10-30 23:17:35 +00:00
// ============= VIDEO ===============
2001-10-29 01:11:18 +00:00
mux_v = aviwrite_new_stream ( muxer , AVIWRITE_TYPE_VIDEO ) ;
2002-02-07 20:22:03 +00:00
mux_v - > buffer_size = 0x200000 ; // 2MB
2001-10-29 01:11:18 +00:00
mux_v - > buffer = malloc ( mux_v - > buffer_size ) ;
mux_v - > source = sh_video ;
mux_v - > h . dwSampleSize = 0 ; // VBR
mux_v - > h . dwScale = 10000 ;
2001-11-01 21:47:28 +00:00
mux_v - > h . dwRate = mux_v - > h . dwScale * ( force_ofps ? force_ofps : sh_video - > fps ) ;
2001-10-29 01:11:18 +00:00
2001-11-03 23:01:17 +00:00
mux_v - > codec = out_video_codec ;
2001-10-30 21:55:28 +00:00
2002-04-11 02:52:03 +00:00
mux_v - > bih = NULL ;
sh_video - > codec = NULL ;
sh_video - > video_out = NULL ;
sh_video - > vfilter = NULL ; // fixme!
2001-10-29 01:11:18 +00:00
switch ( mux_v - > codec ) {
2001-12-08 12:21:23 +00:00
case VCODEC_COPY :
if ( sh_video - > bih )
mux_v - > bih = sh_video - > bih ;
else
{
mux_v - > bih = malloc ( sizeof ( BITMAPINFOHEADER ) ) ;
mux_v - > bih - > biSize = sizeof ( BITMAPINFOHEADER ) ;
mux_v - > bih - > biWidth = sh_video - > disp_w ;
mux_v - > bih - > biHeight = sh_video - > disp_h ;
mux_v - > bih - > biCompression = sh_video - > format ;
mux_v - > bih - > biPlanes = 1 ;
mux_v - > bih - > biBitCount = 24 ; // FIXME!!!
mux_v - > bih - > biSizeImage = mux_v - > bih - > biWidth * mux_v - > bih - > biHeight * ( mux_v - > bih - > biBitCount / 8 ) ;
}
printf ( " videocodec: framecopy (%dx%d %dbpp fourcc=%x) \n " ,
mux_v - > bih - > biWidth , mux_v - > bih - > biHeight ,
mux_v - > bih - > biBitCount , mux_v - > bih - > biCompression ) ;
2001-10-29 01:11:18 +00:00
break ;
2001-12-06 23:59:46 +00:00
case VCODEC_FRAMENO :
mux_v - > bih = malloc ( sizeof ( BITMAPINFOHEADER ) ) ;
mux_v - > bih - > biSize = sizeof ( BITMAPINFOHEADER ) ;
mux_v - > bih - > biWidth = vo_w ;
mux_v - > bih - > biHeight = vo_h ;
mux_v - > bih - > biPlanes = 1 ;
mux_v - > bih - > biBitCount = 24 ;
mux_v - > bih - > biCompression = mmioFOURCC ( ' F ' , ' r ' , ' N ' , ' o ' ) ;
mux_v - > bih - > biSizeImage = mux_v - > bih - > biWidth * mux_v - > bih - > biHeight * ( mux_v - > bih - > biBitCount / 8 ) ;
break ;
2002-04-11 02:52:03 +00:00
default :
2001-12-08 01:43:02 +00:00
2002-04-11 02:52:03 +00:00
switch ( mux_v - > codec ) {
case VCODEC_DIVX4 :
sh_video - > vfilter = vf_open_encoder ( NULL , " divx4 " , mux_v ) ; break ;
case VCODEC_LIBAVCODEC :
sh_video - > vfilter = vf_open_encoder ( NULL , " lavc " , mux_v ) ; break ;
case VCODEC_RAWRGB :
sh_video - > vfilter = vf_open_encoder ( NULL , " rawrgb " , mux_v ) ; break ;
case VCODEC_VFW :
sh_video - > vfilter = vf_open_encoder ( NULL , " vfw " , mux_v ) ; break ;
2002-04-12 21:50:38 +00:00
case VCODEC_LIBDV :
sh_video - > vfilter = vf_open_encoder ( NULL , " libdv " , mux_v ) ; break ;
2002-01-26 20:18:59 +00:00
}
2002-04-11 02:52:03 +00:00
if ( ! mux_v - > bih | | ! sh_video - > vfilter ) {
mp_msg ( MSGT_MENCODER , MSGL_FATAL , " Failed to open the encoder \n " ) ;
mencoder_exit ( 1 , NULL ) ;
2002-01-26 20:18:59 +00:00
}
2002-04-17 21:51:18 +00:00
// append 'expand' filter, it fixes stride problems and renders osd:
sh_video - > vfilter = vf_open_filter ( sh_video - > vfilter , " expand " , " -1:-1:-1:-1:1 " ) ;
2002-04-11 02:52:03 +00:00
sh_video - > vfilter = append_filters ( sh_video - > vfilter ) ;
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " ========================================================================== \n " ) ;
// Go through the codec.conf and find the best codec...
sh_video - > inited = 0 ;
codecs_reset_selection ( 0 ) ;
if ( video_codec ) {
// forced codec by name:
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " Forced video codec: %s \n " , video_codec ) ;
init_video ( sh_video , video_codec , - 1 , - 1 ) ;
} else {
int status ;
// try in stability order: UNTESTED, WORKING, BUGGY, BROKEN
if ( video_family > = 0 ) mp_msg ( MSGT_CPLAYER , MSGL_INFO , MSGTR_TryForceVideoFmt , video_family ) ;
for ( status = CODECS_STATUS__MAX ; status > = CODECS_STATUS__MIN ; - - status ) {
if ( video_family > = 0 ) // try first the preferred codec family:
if ( init_video ( sh_video , NULL , video_family , status ) ) break ;
if ( init_video ( sh_video , NULL , - 1 , status ) ) break ;
}
}
if ( ! sh_video - > inited ) {
mp_msg ( MSGT_CPLAYER , MSGL_ERR , MSGTR_CantFindVideoCodec , sh_video - > format ) ;
mp_msg ( MSGT_CPLAYER , MSGL_HINT , MSGTR_TryUpgradeCodecsConfOrRTFM , get_path ( " codecs.conf " ) ) ;
mencoder_exit ( 1 , NULL ) ;
}
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " %s video codec: [%s] drv:%d prio:%d (%s) \n " ,
video_codec ? mp_gettext ( " Forcing " ) : mp_gettext ( " Detected " ) , sh_video - > codec - > name , sh_video - > codec - > driver , sh_video - > codec - > priority ! = - 1 ? sh_video - > codec - > priority : 0 , sh_video - > codec - > info ) ;
mp_msg ( MSGT_CPLAYER , MSGL_INFO , " ========================================================================== \n " ) ;
2001-12-22 14:32:08 +00:00
}
/* force output fourcc to .. */
2002-01-26 20:18:59 +00:00
if ( ( force_fourcc ! = NULL ) & & ( strlen ( force_fourcc ) > = 4 ) )
2001-12-22 14:32:08 +00:00
{
mux_v - > bih - > biCompression = mmioFOURCC ( force_fourcc [ 0 ] , force_fourcc [ 1 ] ,
force_fourcc [ 2 ] , force_fourcc [ 3 ] ) ;
printf ( " Forcing output fourcc to %x [%.4s] \n " ,
2002-01-26 20:18:59 +00:00
mux_v - > bih - > biCompression , ( char * ) & mux_v - > bih - > biCompression ) ;
2001-10-29 01:11:18 +00:00
}
2002-01-27 02:31:06 +00:00
if ( demuxer - > file_format ! = DEMUXER_TYPE_AVI ) pts_from_bps = 0 ; // it must be 0 for mpeg/asf!
2001-10-30 23:17:35 +00:00
// ============= AUDIO ===============
if ( sh_audio ) {
mux_a = aviwrite_new_stream ( muxer , AVIWRITE_TYPE_AUDIO ) ;
mux_a - > buffer_size = 0x100000 ; //16384;
mux_a - > buffer = malloc ( mux_a - > buffer_size ) ;
mux_a - > source = sh_audio ;
2001-11-03 23:01:17 +00:00
mux_a - > codec = out_audio_codec ;
2001-10-30 23:17:35 +00:00
switch ( mux_a - > codec ) {
2001-12-08 13:30:06 +00:00
case ACODEC_COPY :
2002-01-27 01:23:23 +00:00
if ( sh_audio - > wf ) {
2001-12-08 13:30:06 +00:00
mux_a - > wf = sh_audio - > wf ;
2002-01-27 02:31:06 +00:00
if ( ! sh_audio - > i_bps ) sh_audio - > i_bps = mux_a - > wf - > nAvgBytesPerSec ;
2002-01-27 01:23:23 +00:00
} else {
2001-12-08 13:30:06 +00:00
mux_a - > wf = malloc ( sizeof ( WAVEFORMATEX ) ) ;
2002-01-27 02:31:06 +00:00
mux_a - > wf - > nBlockAlign = 1 ; //mux_a->h.dwSampleSize;
2002-01-27 01:23:23 +00:00
mux_a - > wf - > wFormatTag = sh_audio - > format ;
2001-12-08 13:30:06 +00:00
mux_a - > wf - > nChannels = sh_audio - > channels ;
mux_a - > wf - > nSamplesPerSec = sh_audio - > samplerate ;
2002-01-27 01:23:23 +00:00
mux_a - > wf - > nAvgBytesPerSec = sh_audio - > i_bps ; //mux_a->h.dwSampleSize*mux_a->wf->nSamplesPerSec;
2001-12-13 17:50:19 +00:00
mux_a - > wf - > wBitsPerSample = 16 ; // FIXME
2001-12-08 13:30:06 +00:00
mux_a - > wf - > cbSize = 0 ; // FIXME for l3codeca.acm
}
2002-01-27 02:31:06 +00:00
if ( sh_audio - > audio . dwScale ) {
mux_a - > h . dwSampleSize = sh_audio - > audio . dwSampleSize ;
mux_a - > h . dwScale = sh_audio - > audio . dwScale ;
mux_a - > h . dwRate = sh_audio - > audio . dwRate ;
} else {
mux_a - > h . dwSampleSize = mux_a - > wf - > nBlockAlign ;
mux_a - > h . dwScale = mux_a - > h . dwSampleSize ;
mux_a - > h . dwRate = mux_a - > wf - > nAvgBytesPerSec ;
}
printf ( " audiocodec: framecopy (format=%x chans=%d rate=%d bits=%d bps=%d sample=%d) \n " ,
2001-12-08 13:30:06 +00:00
mux_a - > wf - > wFormatTag , mux_a - > wf - > nChannels , mux_a - > wf - > nSamplesPerSec ,
2002-01-27 02:31:06 +00:00
mux_a - > wf - > wBitsPerSample , mux_a - > wf - > nAvgBytesPerSec , mux_a - > h . dwSampleSize ) ;
2001-10-30 23:17:35 +00:00
break ;
2001-10-31 00:25:28 +00:00
case ACODEC_PCM :
printf ( " CBR PCM audio selected \n " ) ;
mux_a - > h . dwSampleSize = 2 * sh_audio - > channels ;
mux_a - > h . dwScale = 1 ;
mux_a - > h . dwRate = sh_audio - > samplerate ;
mux_a - > wf = malloc ( sizeof ( WAVEFORMATEX ) ) ;
mux_a - > wf - > nBlockAlign = mux_a - > h . dwSampleSize ;
mux_a - > wf - > wFormatTag = 0x1 ; // PCM
mux_a - > wf - > nChannels = sh_audio - > channels ;
mux_a - > wf - > nSamplesPerSec = sh_audio - > samplerate ;
mux_a - > wf - > nAvgBytesPerSec = mux_a - > h . dwSampleSize * mux_a - > wf - > nSamplesPerSec ;
mux_a - > wf - > wBitsPerSample = 16 ;
mux_a - > wf - > cbSize = 0 ; // FIXME for l3codeca.acm
break ;
2001-10-30 23:17:35 +00:00
case ACODEC_VBRMP3 :
2001-12-08 13:30:06 +00:00
printf ( " MP3 audio selected \n " ) ;
2001-10-30 23:17:35 +00:00
mux_a - > h . dwSampleSize = 0 ; // VBR
2002-04-18 17:28:48 +00:00
mux_a - > h . dwRate = force_srate ? force_srate : sh_audio - > samplerate ;
mux_a - > h . dwScale = ( mux_a - > h . dwRate < 32000 ) ? 576 : 1152 ; // samples/frame
2001-11-02 17:43:05 +00:00
if ( sizeof ( MPEGLAYER3WAVEFORMAT ) ! = 30 ) mp_msg ( MSGT_MENCODER , MSGL_WARN , " sizeof(MPEGLAYER3WAVEFORMAT)==%d!=30, maybe broken C compiler? \n " , sizeof ( MPEGLAYER3WAVEFORMAT ) ) ;
mux_a - > wf = malloc ( sizeof ( MPEGLAYER3WAVEFORMAT ) ) ; // should be 30
2001-10-30 23:17:35 +00:00
mux_a - > wf - > wFormatTag = 0x55 ; // MP3
mux_a - > wf - > nChannels = sh_audio - > channels ;
2001-11-02 20:05:36 +00:00
mux_a - > wf - > nSamplesPerSec = force_srate ? force_srate : sh_audio - > samplerate ;
2001-11-02 17:43:05 +00:00
mux_a - > wf - > nAvgBytesPerSec = 192000 / 8 ; // FIXME!
2002-04-18 17:28:48 +00:00
mux_a - > wf - > nBlockAlign = ( mux_a - > h . dwRate < 32000 ) ? 576 : 1152 ; // requires for l3codeca.acm + WMP 6.4
2001-11-02 17:43:05 +00:00
mux_a - > wf - > wBitsPerSample = 0 ; //16;
// from NaNdub: (requires for l3codeca.acm)
mux_a - > wf - > cbSize = 12 ;
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > wID = 1 ;
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > fdwFlags = 2 ;
2002-04-18 17:28:48 +00:00
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > nBlockSize = ( mux_a - > h . dwRate < 32000 ) ? 576 : 1152 ; // ???
2001-11-02 17:43:05 +00:00
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > nFramesPerBlock = 1 ;
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > nCodecDelay = 0 ;
2001-10-30 23:17:35 +00:00
break ;
}
}
2001-11-11 16:15:24 +00:00
printf ( " Writing AVI header... \n " ) ;
2001-10-29 01:11:18 +00:00
aviwrite_write_header ( muxer , muxer_f ) ;
2002-04-11 02:52:03 +00:00
decoded_frameno = 0 ;
2001-10-29 01:11:18 +00:00
2001-10-31 22:59:56 +00:00
if ( sh_audio )
2001-10-31 00:25:28 +00:00
switch ( mux_a - > codec ) {
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-10-31 00:25:28 +00:00
case ACODEC_VBRMP3 :
lame = lame_init ( ) ;
2001-10-31 16:08:01 +00:00
lame_set_bWriteVbrTag ( lame , 0 ) ;
2001-10-31 00:25:28 +00:00
lame_set_in_samplerate ( lame , sh_audio - > samplerate ) ;
lame_set_num_channels ( lame , mux_a - > wf - > nChannels ) ;
2001-11-02 20:05:36 +00:00
lame_set_out_samplerate ( lame , mux_a - > wf - > nSamplesPerSec ) ;
2001-11-02 03:22:33 +00:00
if ( lame_param_vbr ) { // VBR:
lame_set_VBR ( lame , lame_param_vbr ) ; // vbr mode
lame_set_VBR_q ( lame , lame_param_quality + 1 ) ; // 1 = best vbr q 6=~128k
if ( lame_param_br > 0 ) lame_set_VBR_mean_bitrate_kbps ( lame , lame_param_br ) ;
} else { // CBR:
lame_set_quality ( lame , lame_param_quality ) ; // 0 = best q
if ( lame_param_br > 0 ) lame_set_brate ( lame , lame_param_br ) ;
}
if ( lame_param_mode > = 0 ) lame_set_mode ( lame , lame_param_mode ) ; // j-st
if ( lame_param_ratio > 0 ) lame_set_compression_ratio ( lame , lame_param_ratio ) ;
2001-10-31 00:25:28 +00:00
lame_init_params ( lame ) ;
2001-11-02 02:08:00 +00:00
if ( verbose ) {
2001-11-02 03:22:33 +00:00
lame_print_config ( lame ) ;
lame_print_internals ( lame ) ;
2001-11-02 02:08:00 +00:00
}
2001-12-06 22:27:09 +00:00
break ;
# endif
2001-10-31 00:25:28 +00:00
}
2001-10-29 01:11:18 +00:00
signal ( SIGINT , exit_sighandler ) ; // Interrupt from keyboard
signal ( SIGQUIT , exit_sighandler ) ; // Quit from keyboard
signal ( SIGTERM , exit_sighandler ) ; // kill
2002-01-27 18:29:33 +00:00
timer_start = GetTimerMS ( ) ;
2002-02-10 00:07:34 +00:00
if ( seek_to_sec ) {
int a , b ; float d ;
if ( sscanf ( seek_to_sec , " %d:%d:%f " , & a , & b , & d ) = = 3 )
d + = 3600 * a + 60 * b ;
else if ( sscanf ( seek_to_sec , " %d:%f " , & a , & d ) = = 2 )
d + = 60 * a ;
else
sscanf ( seek_to_sec , " %f " , & d ) ;
demux_seek ( demuxer , d , 1 ) ;
}
2002-04-12 10:40:38 +00:00
if ( tv_param_on = = 1 )
{
fprintf ( stderr , " Forcing audio preload to 0, max pts correction to 0 \n " ) ;
audio_preload = 0.0 ;
default_max_pts_correction = 0 ;
}
2001-10-29 01:11:18 +00:00
while ( ! eof ) {
2001-10-30 21:22:28 +00:00
float frame_time = 0 ;
2001-10-29 01:11:18 +00:00
int blit_frame = 0 ;
float a_pts = 0 ;
float v_pts = 0 ;
2001-10-30 21:55:28 +00:00
unsigned char * start = NULL ;
int in_size ;
2001-11-01 21:47:28 +00:00
int skip_flag = 0 ; // 1=skip -1=duplicate
2001-10-29 01:11:18 +00:00
2002-04-12 10:40:38 +00:00
if ( ( end_at_type = = END_AT_SIZE & & end_at < = ftello ( muxer_f ) ) | |
2002-01-15 00:20:50 +00:00
( end_at_type = = END_AT_TIME & & end_at < sh_video - > timer ) )
break ;
2001-11-03 00:44:02 +00:00
if ( play_n_frames > = 0 ) {
- - play_n_frames ;
if ( play_n_frames < 0 ) break ;
}
2001-10-30 23:17:35 +00:00
if ( sh_audio ) {
// get audio:
2001-10-31 00:25:28 +00:00
while ( mux_a - > timer - audio_preload < mux_v - > timer ) {
2001-11-03 21:00:04 +00:00
int len = 0 ;
2002-04-12 10:40:38 +00:00
ptimer_start = GetTimerMS ( ) ;
2001-10-30 23:17:35 +00:00
if ( mux_a - > h . dwSampleSize ) {
2001-11-01 02:31:23 +00:00
// CBR - copy 0.5 sec of audio
2001-10-31 00:25:28 +00:00
switch ( mux_a - > codec ) {
2001-12-08 13:30:06 +00:00
case ACODEC_COPY : // copy
2002-01-27 02:31:06 +00:00
len = mux_a - > wf - > nAvgBytesPerSec / 2 ;
2001-10-31 00:25:28 +00:00
len / = mux_a - > h . dwSampleSize ; if ( len < 1 ) len = 1 ;
len * = mux_a - > h . dwSampleSize ;
len = demux_read_data ( sh_audio - > ds , mux_a - > buffer , len ) ;
break ;
case ACODEC_PCM :
len = mux_a - > h . dwSampleSize * ( mux_a - > h . dwRate / 2 ) ;
2001-10-31 16:08:01 +00:00
len = dec_audio ( sh_audio , mux_a - > buffer , len ) ;
2001-10-31 00:25:28 +00:00
break ;
}
2001-10-30 23:17:35 +00:00
} else {
2001-11-01 02:31:23 +00:00
// VBR - encode/copy an audio frame
switch ( mux_a - > codec ) {
2001-12-08 13:30:06 +00:00
case ACODEC_COPY : // copy
2002-01-26 22:30:02 +00:00
len = ds_get_packet ( sh_audio - > ds , ( unsigned char * * ) & mux_a - > buffer ) ;
// printf("VBR audio framecopy not yet implemented!\n");
2001-11-01 02:31:23 +00:00
break ;
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
2001-11-01 02:31:23 +00:00
case ACODEC_VBRMP3 :
2001-10-31 16:08:01 +00:00
while ( mux_a - > buffer_len < 4 ) {
unsigned char tmp [ 2304 ] ;
int len = dec_audio ( sh_audio , tmp , 2304 ) ;
if ( len < = 0 ) break ; // eof
len = lame_encode_buffer_interleaved ( lame ,
tmp , len / 4 ,
mux_a - > buffer + mux_a - > buffer_len , mux_a - > buffer_size - mux_a - > buffer_len ) ;
if ( len < 0 ) break ; // error
mux_a - > buffer_len + = len ;
}
if ( mux_a - > buffer_len < 4 ) break ;
len = mp_decode_mp3_header ( mux_a - > buffer ) ;
2001-11-02 20:05:36 +00:00
//printf("%d\n",len);
2001-10-31 16:08:01 +00:00
if ( len < = 0 ) break ; // bad frame!
2002-04-18 17:28:48 +00:00
// printf("[%d]\n",mp_mp3_get_lsf(mux_a->buffer));
2001-10-31 16:08:01 +00:00
while ( mux_a - > buffer_len < len ) {
unsigned char tmp [ 2304 ] ;
int len = dec_audio ( sh_audio , tmp , 2304 ) ;
if ( len < = 0 ) break ; // eof
len = lame_encode_buffer_interleaved ( lame ,
tmp , len / 4 ,
mux_a - > buffer + mux_a - > buffer_len , mux_a - > buffer_size - mux_a - > buffer_len ) ;
if ( len < 0 ) break ; // error
mux_a - > buffer_len + = len ;
}
2001-11-01 02:31:23 +00:00
break ;
2001-12-06 22:27:09 +00:00
# endif
2001-11-01 02:31:23 +00:00
}
2001-10-30 23:17:35 +00:00
}
2001-10-31 00:25:28 +00:00
if ( len < = 0 ) break ; // EOF?
aviwrite_write_chunk ( muxer , mux_a , muxer_f , len , 0 ) ;
2001-11-03 21:09:21 +00:00
if ( ! mux_a - > h . dwSampleSize & & mux_a - > timer > 0 )
2001-12-06 21:50:35 +00:00
mux_a - > wf - > nAvgBytesPerSec = 0.5f + ( double ) mux_a - > size / mux_a - > timer ; // avg bps (VBR)
2001-10-31 16:08:01 +00:00
if ( mux_a - > buffer_len > = len ) {
mux_a - > buffer_len - = len ;
memcpy ( mux_a - > buffer , mux_a - > buffer + len , mux_a - > buffer_len ) ;
}
2002-04-12 10:40:38 +00:00
audiosamples + + ;
audiorate + = ( GetTimerMS ( ) - ptimer_start ) ;
2001-10-30 23:17:35 +00:00
}
}
// get video frame!
2002-04-12 10:40:38 +00:00
2001-10-30 23:17:35 +00:00
in_size = video_read_frame ( sh_video , & frame_time , & start , force_fps ) ;
if ( in_size < 0 ) { eof = 1 ; break ; }
2001-12-06 23:59:46 +00:00
sh_video - > timer + = frame_time ; + + decoded_frameno ;
2002-04-12 10:40:38 +00:00
2001-11-01 21:47:28 +00:00
v_timer_corr - = frame_time - ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ;
2002-04-11 02:52:03 +00:00
if ( demuxer2 ) { // 3-pass encoding, read control file (frameno.avi)
2002-01-26 22:30:02 +00:00
// find our frame:
while ( next_frameno < decoded_frameno ) {
int * start ;
int len = ds_get_packet ( demuxer2 - > video , ( unsigned char * * ) & start ) ;
if ( len < 0 ) { eof = 1 ; break ; }
if ( len = = 0 ) - - skip_flag ; else // duplicate
if ( len = = 4 ) next_frameno = start [ 0 ] ;
}
if ( eof ) break ;
2002-04-12 10:40:38 +00:00
// if(skip_flag) printf("!!!!!!!!!!!!\n");
2002-01-26 22:30:02 +00:00
skip_flag = next_frameno - decoded_frameno ;
// find next frame:
while ( next_frameno < = decoded_frameno ) {
int * start ;
int len = ds_get_packet ( demuxer2 - > video , ( unsigned char * * ) & start ) ;
if ( len < 0 ) { eof = 1 ; break ; }
if ( len = = 0 ) - - skip_flag ; else // duplicate
if ( len = = 4 ) next_frameno = start [ 0 ] ;
}
// if(eof) break;
// printf("Current fno=%d requested=%d skip=%d \n",decoded_frameno,fno,skip_flag);
} else {
2001-11-01 21:47:28 +00:00
// check frame duplicate/drop:
2002-04-15 02:33:05 +00:00
if ( v_timer_corr > = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate & &
( skip_limit < 0 | | skip_flag < skip_limit ) ) {
2001-11-01 21:47:28 +00:00
v_timer_corr - = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ;
+ + skip_flag ; // skip
} else
2002-04-15 02:33:05 +00:00
while ( v_timer_corr < = - ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate & &
( skip_limit < 0 | | ( - skip_flag ) < skip_limit ) ) {
2001-11-01 21:47:28 +00:00
v_timer_corr + = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ;
- - skip_flag ; // dup
}
while ( ( v_pts_corr < = - ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate & & skip_flag > 0 )
| | ( v_pts_corr < = - 2 * ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ) ) {
v_pts_corr + = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ;
- - skip_flag ; // dup
}
if ( ( v_pts_corr > = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate & & skip_flag < 0 )
| | ( v_pts_corr > = 2 * ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ) )
if ( skip_flag < = 0 ) { // we can't skip more than 1 frame now
v_pts_corr - = ( float ) mux_v - > h . dwScale / mux_v - > h . dwRate ;
+ + skip_flag ; // skip
}
2001-10-29 01:11:18 +00:00
2002-01-26 22:30:02 +00:00
} // demuxer2
2002-04-12 10:40:38 +00:00
ptimer_start = GetTimerMS ( ) ;
2001-10-29 01:11:18 +00:00
switch ( mux_v - > codec ) {
2001-12-08 12:21:23 +00:00
case VCODEC_COPY :
2001-10-30 21:55:28 +00:00
mux_v - > buffer = start ;
2001-11-02 20:05:36 +00:00
if ( skip_flag < = 0 ) aviwrite_write_chunk ( muxer , mux_v , muxer_f , in_size , ( sh_video - > ds - > flags & 1 ) ? 0x10 : 0 ) ;
2001-10-30 21:55:28 +00:00
break ;
2002-04-11 02:52:03 +00:00
case VCODEC_FRAMENO :
2001-12-06 23:59:46 +00:00
mux_v - > buffer = & decoded_frameno ; // tricky
2001-12-07 00:12:51 +00:00
if ( skip_flag < = 0 ) aviwrite_write_chunk ( muxer , mux_v , muxer_f , sizeof ( int ) , 0x10 ) ;
2002-02-07 20:22:03 +00:00
break ;
2002-04-11 02:52:03 +00:00
default :
// decode_video will callback down to ve_*.c encoders, through the video filters
blit_frame = decode_video ( sh_video , start , in_size , ( skip_flag > 0 ) ? 1 : 0 ) ;
2001-11-02 20:05:36 +00:00
if ( skip_flag > 0 ) break ;
2002-04-11 02:52:03 +00:00
if ( ! blit_frame ) aviwrite_write_chunk ( muxer , mux_v , muxer_f , 0 , 0 ) ; // empty.
2001-10-29 01:11:18 +00:00
}
2001-11-01 21:47:28 +00:00
2002-04-12 10:40:38 +00:00
videosamples + + ;
videorate + = ( GetTimerMS ( ) - ptimer_start ) ;
2001-11-01 21:47:28 +00:00
if ( skip_flag < 0 ) {
2001-11-01 02:31:23 +00:00
// duplicate frame
2002-04-12 10:40:38 +00:00
if ( ! tv_param_on & & ! verbose ) printf ( " \n duplicate %d frame(s)!!! \n " , - skip_flag ) ;
2001-11-01 21:47:28 +00:00
while ( skip_flag < 0 ) {
2002-04-12 10:40:38 +00:00
duplicatedframes + + ;
2001-11-01 21:47:28 +00:00
aviwrite_write_chunk ( muxer , mux_v , muxer_f , 0 , 0 ) ;
+ + skip_flag ;
}
2001-11-02 20:05:36 +00:00
} else
if ( skip_flag > 0 ) {
2001-11-01 02:31:23 +00:00
// skip frame
2002-04-12 10:40:38 +00:00
if ( ! tv_param_on & & ! verbose ) printf ( " \n skip frame!!! \n " ) ;
skippedframes + + ;
2001-11-01 21:47:28 +00:00
- - skip_flag ;
2001-11-01 02:31:23 +00:00
}
2002-01-26 22:30:02 +00:00
if ( sh_audio & & ! demuxer2 ) {
2001-11-01 02:31:23 +00:00
float AV_delay , x ;
// A-V sync!
if ( pts_from_bps ) {
unsigned int samples = ( sh_audio - > audio . dwSampleSize ) ?
( ( ds_tell ( d_audio ) - sh_audio - > a_in_buffer_len ) / sh_audio - > audio . dwSampleSize ) :
( d_audio - > pack_no ) ; // <- used for VBR audio
2002-01-27 14:35:48 +00:00
// printf("samples=%d \n",samples);
2001-11-01 02:31:23 +00:00
a_pts = samples * ( float ) sh_audio - > audio . dwScale / ( float ) sh_audio - > audio . dwRate ;
delay_corrected = 1 ;
} else {
// PTS = (last timestamp) + (bytes after last timestamp)/(bytes per sec)
a_pts = d_audio - > pts ;
if ( ! delay_corrected ) if ( a_pts ) delay_corrected = 1 ;
//printf("*** %5.3f ***\n",a_pts);
a_pts + = ( ds_tell_pts ( d_audio ) - sh_audio - > a_in_buffer_len ) / ( float ) sh_audio - > i_bps ;
}
v_pts = d_video - > pts ;
// av = compensated (with out buffering delay) A-V diff
2001-11-01 21:47:28 +00:00
AV_delay = ( a_pts - v_pts ) ; AV_delay - = mux_a - > timer - ( mux_v - > timer - ( v_timer_corr + v_pts_corr ) ) ;
2001-11-01 02:31:23 +00:00
// compensate input video timer by av:
x = AV_delay * 0.1f ;
if ( x < - max_pts_correction ) x = - max_pts_correction ; else
if ( x > max_pts_correction ) x = max_pts_correction ;
if ( default_max_pts_correction > = 0 )
max_pts_correction = default_max_pts_correction ;
else
max_pts_correction = sh_video - > frametime * 0.10 ; // +-10% of time
// sh_video->timer-=x;
c_total + = x ;
2001-11-01 21:47:28 +00:00
v_pts_corr + = x ;
2001-11-01 02:31:23 +00:00
}
2002-01-27 18:29:33 +00:00
// printf("A:%6.1f V:%6.1f A-V:%7.3f oAV:%7.3f diff:%7.3f ct:%7.3f vpc:%7.3f \r",
// a_pts,v_pts,a_pts-v_pts,
// (float)(mux_a->timer-mux_v->timer),
// AV_delay, c_total, v_pts_corr );
// printf("V:%6.1f \r", d_video->pts );
2001-11-01 02:31:23 +00:00
#if 0
mp_msg ( MSGT_AVSYNC , MSGL_STATUS , " A:%6.1f V:%6.1f A-V:%7.3f ct:%7.3f %3d/%3d %2d%% %2d%% %4.1f%% %d%% \r " ,
a_pts , v_pts , a_pts - v_pts , c_total ,
( int ) sh_video - > num_frames , ( int ) sh_video - > num_frames_decoded ,
( sh_video - > timer > 0.5 ) ? ( int ) ( 100.0 * video_time_usage / ( double ) sh_video - > timer ) : 0 ,
( sh_video - > timer > 0.5 ) ? ( int ) ( 100.0 * vout_time_usage / ( double ) sh_video - > timer ) : 0 ,
( sh_video - > timer > 0.5 ) ? ( 100.0 * audio_time_usage / ( double ) sh_video - > timer ) : 0
, cache_fill_status
) ;
# endif
2002-01-27 18:29:33 +00:00
{ float t = ( GetTimerMS ( ) - timer_start ) * 0.001f ;
float len = ( demuxer - > movi_end - demuxer - > movi_start ) ;
float p = len > 1000 ? ( float ) ( demuxer - > filepos - demuxer - > movi_start ) / len : 0 ;
2002-01-27 21:37:46 +00:00
if ( ! len & & sh_audio & & sh_audio - > audio . dwLength > 100 ) {
p = ( sh_audio - > audio . dwSampleSize ? ds_tell ( sh_audio - > ds ) / sh_audio - > audio . dwSampleSize : sh_audio - > ds - > pack_no )
/ ( float ) ( sh_audio - > audio . dwLength ) ;
}
2002-01-27 20:34:31 +00:00
#if 0
mp_msg ( MSGT_AVSYNC , MSGL_STATUS , " %d < %d < %d \r " ,
( int ) demuxer - > movi_start ,
( int ) demuxer - > filepos ,
( int ) demuxer - > movi_end ) ;
# else
2002-04-12 10:40:38 +00:00
if ( verbose ) {
mp_msg ( MSGT_AVSYNC , MSGL_STATUS , " Pos:%6.1fs %6df (%2d%%) %3dfps Trem:%4dmin %3dmb A-V:%5.3f [%d:%d] A/Vms %d/%d D/S %d/%d \r " ,
mux_v - > timer , decoded_frameno , ( int ) ( p * 100 ) ,
( t > 1 ) ? ( int ) ( decoded_frameno / t ) : 0 ,
( p > 0.001 ) ? ( int ) ( ( t / p - t ) / 60 ) : 0 ,
( p > 0.001 ) ? ( int ) ( ftello ( muxer_f ) / p / 1024 / 1024 ) : 0 ,
v_pts_corr ,
( mux_v - > timer > 1 ) ? ( int ) ( mux_v - > size / mux_v - > timer / 125 ) : 0 ,
( mux_a & & mux_a - > timer > 1 ) ? ( int ) ( mux_a - > size / mux_a - > timer / 125 ) : 0 ,
audiorate / audiosamples , videorate / videosamples ,
duplicatedframes , skippedframes
) ;
} else
2002-01-27 21:46:06 +00:00
mp_msg ( MSGT_AVSYNC , MSGL_STATUS , " Pos:%6.1fs %6df (%2d%%) %3dfps Trem:%4dmin %3dmb A-V:%5.3f [%d:%d] \r " ,
2002-01-27 18:29:33 +00:00
mux_v - > timer , decoded_frameno , ( int ) ( p * 100 ) ,
( t > 1 ) ? ( int ) ( decoded_frameno / t ) : 0 ,
2002-01-27 20:34:31 +00:00
( p > 0.001 ) ? ( int ) ( ( t / p - t ) / 60 ) : 0 ,
2002-01-27 18:29:33 +00:00
( p > 0.001 ) ? ( int ) ( ftell ( muxer_f ) / p / 1024 / 1024 ) : 0 ,
2002-01-27 21:46:06 +00:00
v_pts_corr ,
( mux_v - > timer > 1 ) ? ( int ) ( mux_v - > size / mux_v - > timer / 125 ) : 0 ,
2002-01-31 00:38:53 +00:00
( mux_a & & mux_a - > timer > 1 ) ? ( int ) ( mux_a - > size / mux_a - > timer / 125 ) : 0
2002-01-27 18:29:33 +00:00
) ;
2002-01-27 20:34:31 +00:00
# endif
2002-01-27 18:29:33 +00:00
}
2001-11-01 02:31:23 +00:00
fflush ( stdout ) ;
2002-04-17 21:51:18 +00:00
# ifdef USE_SUB
// find sub
if ( subtitles & & d_video - > pts > 0 ) {
float pts = d_video - > pts ;
if ( sub_fps = = 0 ) sub_fps = sh_video - > fps ;
if ( pts > sub_last_pts | | pts < sub_last_pts - 1.0 ) {
find_sub ( subtitles , sub_uses_time ? ( 100 * ( pts + sub_delay ) ) : ( ( pts + sub_delay ) * sub_fps ) ) ; // FIXME! frame counter...
sub_last_pts = pts ;
}
}
# endif
# ifdef USE_DVDREAD
// DVD sub:
if ( vo_spudec ) {
unsigned char * packet = NULL ;
int len ;
while ( ( len = ds_get_packet_sub ( d_dvdsub , & packet ) ) > 0 ) {
mp_msg ( MSGT_MENCODER , MSGL_V , " \r DVD sub: len=%d v_pts=%5.3f s_pts=%5.3f \n " , len , d_video - > pts , d_dvdsub - > pts ) ;
spudec_assemble ( vo_spudec , packet , len , 90000 * d_dvdsub - > pts ) ;
}
spudec_heartbeat ( vo_spudec , 90000 * d_video - > pts ) ;
2002-04-17 22:17:32 +00:00
vo_osd_changed ( OSDTYPE_SPU ) ;
2002-04-17 21:51:18 +00:00
}
# endif
2001-10-29 01:11:18 +00:00
} // while(!eof)
2001-12-06 22:27:09 +00:00
# ifdef HAVE_MP3LAME
// fixup CBR mp3 audio header:
2001-12-06 21:50:35 +00:00
if ( sh_audio & & mux_a - > codec = = ACODEC_VBRMP3 & & ! lame_param_vbr ) {
mux_a - > h . dwSampleSize = 1 ;
2002-04-18 14:23:07 +00:00
( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > nBlockSize =
( mux_a - > size + ( mux_a - > h . dwLength > > 1 ) ) / mux_a - > h . dwLength ;
mux_a - > h . dwLength = mux_a - > size ;
2001-12-06 21:50:35 +00:00
mux_a - > h . dwRate = mux_a - > wf - > nAvgBytesPerSec ;
mux_a - > h . dwScale = 1 ;
2002-04-18 14:23:07 +00:00
mux_a - > wf - > nBlockAlign = 1 ;
printf ( " \n \n CBR audio: %d bytes/sec, %d bytes/block \n " ,
mux_a - > h . dwRate , ( ( MPEGLAYER3WAVEFORMAT * ) ( mux_a - > wf ) ) - > nBlockSize ) ;
2001-12-06 21:50:35 +00:00
}
2001-12-06 22:27:09 +00:00
# endif
2001-12-06 21:50:35 +00:00
2001-11-11 16:15:24 +00:00
printf ( " \n Writing AVI index... \n " ) ;
2001-10-29 01:11:18 +00:00
aviwrite_write_index ( muxer , muxer_f ) ;
2002-04-12 10:40:38 +00:00
muxer_f_size = ftello ( muxer_f ) ;
2001-11-02 02:08:00 +00:00
printf ( " Fixup AVI header... \n " ) ;
2001-10-29 01:11:18 +00:00
fseek ( muxer_f , 0 , SEEK_SET ) ;
aviwrite_write_header ( muxer , muxer_f ) ; // update header
2002-01-27 14:16:32 +00:00
fclose ( muxer_f ) ;
2002-01-27 00:43:57 +00:00
if ( out_video_codec = = VCODEC_FRAMENO & & mux_v - > timer > 100 ) {
2002-01-27 20:34:31 +00:00
printf ( " Recommended video bitrate for 650MB CD: %d \n " , ( int ) ( ( 650 * 1024 * 1024 - muxer_f_size ) / mux_v - > timer / 125 ) ) ;
printf ( " Recommended video bitrate for 700MB CD: %d \n " , ( int ) ( ( 700 * 1024 * 1024 - muxer_f_size ) / mux_v - > timer / 125 ) ) ;
printf ( " Recommended video bitrate for 800MB CD: %d \n " , ( int ) ( ( 800 * 1024 * 1024 - muxer_f_size ) / mux_v - > timer / 125 ) ) ;
2002-01-27 00:43:57 +00:00
}
2002-01-26 16:32:06 +00:00
printf ( " \n Video stream: %8.3f kbit/s (%d bps) size: %d bytes %5.3f secs %d frames \n " ,
2002-04-12 10:40:38 +00:00
( float ) ( mux_v - > size / mux_v - > timer * 8.0f / 1000.0f ) , ( int ) ( mux_v - > size / mux_v - > timer ) , ( int ) mux_v - > size , ( float ) mux_v - > timer , decoded_frameno ) ;
2001-12-06 23:59:46 +00:00
if ( sh_audio )
printf ( " \n Audio stream: %8.3f kbit/s (%d bps) size: %d bytes %5.3f secs \n " ,
2002-04-12 10:40:38 +00:00
( float ) ( mux_a - > size / mux_a - > timer * 8.0f / 1000.0f ) , ( int ) ( mux_a - > size / mux_a - > timer ) , ( int ) mux_a - > size , ( float ) mux_a - > timer ) ;
2001-12-06 23:59:46 +00:00
2001-11-02 01:25:13 +00:00
if ( stream ) free_stream ( stream ) ; // kill cache thread
2001-12-04 20:35:31 +00:00
return interrupted ;
2001-10-29 01:11:18 +00:00
}
2002-01-15 00:20:50 +00:00
static int parse_end_at ( struct config * conf , const char * param )
{
int i ;
end_at_type = END_AT_NONE ;
/* End at size parsing */
{
char unit [ 4 ] ;
end_at_type = END_AT_SIZE ;
if ( sscanf ( param , " %d%3s " , & end_at , unit ) = = 2 ) {
if ( ! strcasecmp ( unit , " b " ) )
;
else if ( ! strcasecmp ( unit , " kb " ) )
end_at * = 1024 ;
else if ( ! strcasecmp ( unit , " mb " ) )
end_at * = 1024 * 1024 ;
else
end_at_type = END_AT_NONE ;
}
else
end_at_type = END_AT_NONE ;
}
/* End at time parsing. This has to be last because of
* sscanf ( " %f " , . . . ) below */
if ( end_at_type = = END_AT_NONE )
{
int a , b ; float d ;
end_at_type = END_AT_TIME ;
if ( sscanf ( param , " %d:%d:%f " , & a , & b , & d ) = = 3 )
end_at = 3600 * a + 60 * b + d ;
else if ( sscanf ( param , " %d:%f " , & a , & d ) = = 2 )
end_at = 60 * a + d ;
else if ( sscanf ( param , " %f " , & d ) = = 1 )
end_at = d ;
else
end_at_type = END_AT_NONE ;
}
if ( end_at_type = = END_AT_NONE )
return ERR_FUNC_ERR ;
return 1 ;
}
2002-01-31 00:38:53 +00:00
/* Flip the image in src and store the result in dst. src and dst may overlap.
width is the size of each line in bytes . */
static uint8_t * flip_upside_down ( uint8_t * dst , const uint8_t * src , int width ,
int height )
{
uint8_t * tmp = malloc ( width ) ;
int i ;
for ( i = 0 ; i < height / 2 ; i + + ) {
memcpy ( tmp , & src [ i * width ] , width ) ;
memcpy ( & dst [ i * width ] , & src [ ( height - i ) * width ] , width ) ;
memcpy ( & dst [ ( height - i ) * width ] , tmp , width ) ;
}
free ( tmp ) ;
return dst ;
}
2002-03-23 13:13:12 +00:00