1 /* $Id: test.c,v 1.82 2005/11/19 08:25:54 titer Exp $
\r
3 This file is part of the HandBrake source code.
\r
4 Homepage: <http://handbrake.m0k.org/>.
\r
5 It may be used under the terms of the GNU General Public License. */
\r
9 #include <sys/time.h>
\r
16 static int debug = HB_DEBUG_NONE;
\r
17 static int update = 0;
\r
18 static char * input = NULL;
\r
19 static char * output = NULL;
\r
20 static char * format = NULL;
\r
21 static int titleindex = 1;
\r
22 static int twoPass = 0;
\r
23 static int deinterlace = 0;
\r
24 static int grayscale = 0;
\r
25 static int vcodec = HB_VCODEC_FFMPEG;
\r
26 static int h264_13 = 0;
\r
27 static int h264_30 = 0;
\r
28 static char * audios = NULL;
\r
29 static int surround = 0;
\r
31 static int width = 0;
\r
32 static int height = 0;
\r
33 static int crop[4] = { -1,-1,-1,-1 };
\r
35 static int vrate = 0;
\r
36 static int arate = 0;
\r
37 static float vquality = -1.0;
\r
38 static int vbitrate = 0;
\r
39 static int size = 0;
\r
40 static int abitrate = 0;
\r
42 static int acodec = 0;
\r
43 static int pixelratio = 0;
\r
44 static int chapter_start = 0;
\r
45 static int chapter_end = 0;
\r
46 static int chapter_markers = 0;
\r
48 static char *x264opts = NULL;
\r
49 static char *x264opts2 = NULL;
\r
50 static int maxHeight = 0;
\r
51 static int maxWidth = 0;
\r
53 /* Exit cleanly on Ctrl-C */
\r
54 static volatile int die = 0;
\r
55 static void SigHandler( int );
\r
58 static void ShowCommands();
\r
59 static void ShowHelp();
\r
60 static int ParseOptions( int argc, char ** argv );
\r
61 static int CheckOptions( int argc, char ** argv );
\r
62 static int HandleEvents( hb_handle_t * h );
\r
64 int main( int argc, char ** argv )
\r
70 /* Parse command line */
\r
71 if( ParseOptions( argc, argv ) ||
\r
72 CheckOptions( argc, argv ) )
\r
78 h = hb_init( debug, update );
\r
81 fprintf( stderr, "HandBrake %s (%d) - http://handbrake.m0k.org/\n",
\r
82 hb_get_version( h ), hb_get_build( h ) );
\r
84 /* Check for update */
\r
87 if( ( build = hb_check_update( h, &version ) ) > -1 )
\r
89 fprintf( stderr, "You are using an old version of "
\r
90 "HandBrake.\nLatest is %s (build %d).\n", version,
\r
95 fprintf( stderr, "Your version of HandBrake is up to "
\r
103 fprintf( stderr, "%d CPU%s detected\n", hb_get_cpu_count(),
\r
104 hb_get_cpu_count( h ) > 1 ? "s" : "" );
\r
107 fprintf( stderr, "Forcing %d CPU%s\n", cpu,
\r
108 cpu > 1 ? "s" : "" );
\r
109 hb_set_cpu_count( h, cpu );
\r
112 /* Exit ASAP on Ctrl-C */
\r
113 signal( SIGINT, SigHandler );
\r
115 /* Feed libhb with a DVD to scan */
\r
116 fprintf( stderr, "Opening %s...\n", input );
\r
117 hb_scan( h, input, titleindex );
\r
122 #if !defined(SYS_BEOS)
\r
129 tv.tv_usec = 100000;
\r
132 FD_SET( STDIN_FILENO, &fds );
\r
133 ret = select( STDIN_FILENO + 1, &fds, NULL, NULL, &tv );
\r
139 while( size < 256 &&
\r
140 read( STDIN_FILENO, &buf[size], 1 ) > 0 )
\r
142 if( buf[size] == '\n' )
\r
149 if( size >= 256 || buf[size] == '\n' )
\r
178 if( input ) free( input );
\r
179 if( output ) free( output );
\r
180 if( format ) free( format );
\r
181 if( audios ) free( audios );
\r
182 if( x264opts ) free (x264opts );
\r
183 if( x264opts2 ) free (x264opts2 );
\r
185 fprintf( stderr, "HandBrake has exited.\n" );
\r
190 static void ShowCommands()
\r
192 fprintf( stderr, "Commands:\n" );
\r
193 fprintf( stderr, " [h]elp Show this message\n" );
\r
194 fprintf( stderr, " [q]uit Exit HandBrakeCLI\n" );
\r
195 fprintf( stderr, " [p]ause Pause encoding\n" );
\r
196 fprintf( stderr, " [r]esume Resume encoding\n" );
\r
199 static void PrintTitleInfo( hb_title_t * title )
\r
201 hb_chapter_t * chapter;
\r
202 hb_audio_t * audio;
\r
203 hb_subtitle_t * subtitle;
\r
206 fprintf( stderr, "+ title %d:\n", title->index );
\r
207 fprintf( stderr, " + vts %d, ttn %d, cells %d->%d (%d blocks)\n",
\r
208 title->vts, title->ttn, title->cell_start, title->cell_end,
\r
209 title->block_count );
\r
210 fprintf( stderr, " + duration: %02d:%02d:%02d\n",
\r
211 title->hours, title->minutes, title->seconds );
\r
212 fprintf( stderr, " + size: %dx%d, aspect: %.2f, %.3f fps\n",
\r
213 title->width, title->height,
\r
214 (float) title->aspect / HB_ASPECT_BASE,
\r
215 (float) title->rate / title->rate_base );
\r
216 fprintf( stderr, " + autocrop: %d/%d/%d/%d\n", title->crop[0],
\r
217 title->crop[1], title->crop[2], title->crop[3] );
\r
218 fprintf( stderr, " + chapters:\n" );
\r
219 for( i = 0; i < hb_list_count( title->list_chapter ); i++ )
\r
221 chapter = hb_list_item( title->list_chapter, i );
\r
222 fprintf( stderr, " + %d: cells %d->%d, %d blocks, duration "
\r
223 "%02d:%02d:%02d\n", chapter->index,
\r
224 chapter->cell_start, chapter->cell_end,
\r
225 chapter->block_count, chapter->hours, chapter->minutes,
\r
226 chapter->seconds );
\r
228 fprintf( stderr, " + audio tracks:\n" );
\r
229 for( i = 0; i < hb_list_count( title->list_audio ); i++ )
\r
231 audio = hb_list_item( title->list_audio, i );
\r
232 if( audio->codec & HB_ACODEC_AC3 )
\r
234 fprintf( stderr, " + %d, %s, %dHz, %dbps\n", i + 1,
\r
235 audio->lang, audio->rate, audio->bitrate );
\r
239 fprintf( stderr, " + %d, %s\n", i + 1, audio->lang );
\r
242 fprintf( stderr, " + subtitle tracks:\n" );
\r
243 for( i = 0; i < hb_list_count( title->list_subtitle ); i++ )
\r
245 subtitle = hb_list_item( title->list_subtitle, i );
\r
246 fprintf( stderr, " + %d, %s\n", i + 1, subtitle->lang );
\r
250 static int HandleEvents( hb_handle_t * h )
\r
253 hb_get_state( h, &s );
\r
256 case HB_STATE_IDLE:
\r
257 /* Nothing to do */
\r
260 #define p s.param.scanning
\r
261 case HB_STATE_SCANNING:
\r
262 /* Show what title is currently being scanned */
\r
263 fprintf( stderr, "Scanning title %d", p.title_cur );
\r
265 fprintf( stderr, " of %d", p.title_count );
\r
266 fprintf( stderr, "...\n" );
\r
270 case HB_STATE_SCANDONE:
\r
273 hb_title_t * title;
\r
276 list = hb_get_titles( h );
\r
278 if( !hb_list_count( list ) )
\r
280 /* No valid title, stop right there */
\r
281 fprintf( stderr, "No title found.\n" );
\r
287 /* Scan-only mode, print infos and exit */
\r
289 for( i = 0; i < hb_list_count( list ); i++ )
\r
291 title = hb_list_item( list, i );
\r
292 PrintTitleInfo( title );
\r
298 /* Set job settings */
\r
299 title = hb_list_item( list, 0 );
\r
302 PrintTitleInfo( title );
\r
304 if( chapter_start && chapter_end )
\r
306 job->chapter_start = MAX( job->chapter_start,
\r
308 job->chapter_end = MIN( job->chapter_end,
\r
310 job->chapter_end = MAX( job->chapter_start,
\r
311 job->chapter_end );
\r
314 if ( chapter_markers )
\r
316 job->chapter_markers = chapter_markers;
\r
319 if( crop[0] >= 0 && crop[1] >= 0 &&
\r
320 crop[2] >= 0 && crop[3] >= 0 )
\r
322 memcpy( job->crop, crop, 4 * sizeof( int ) );
\r
325 job->deinterlace = deinterlace;
\r
326 job->grayscale = grayscale;
\r
327 job->pixel_ratio = pixelratio;
\r
329 if( width && height )
\r
331 job->width = width;
\r
332 job->height = height;
\r
336 job->width = width;
\r
337 hb_fix_aspect( job, HB_KEEP_WIDTH );
\r
341 job->height = height;
\r
342 hb_fix_aspect( job, HB_KEEP_HEIGHT );
\r
344 else if( !width && !height && !pixelratio )
\r
346 hb_fix_aspect( job, HB_KEEP_WIDTH );
\r
349 if( vquality >= 0.0 && vquality <= 1.0 )
\r
351 job->vquality = vquality;
\r
354 else if( vbitrate )
\r
356 job->vquality = -1.0;
\r
357 job->vbitrate = vbitrate;
\r
361 job->vcodec = vcodec;
\r
365 job->h264_level = 13;
\r
369 job->h264_level = 30;
\r
373 job->vrate = 27000000;
\r
374 job->vrate_base = vrate;
\r
378 job->arate = arate;
\r
383 if( strcasecmp( audios, "none" ) )
\r
385 int audio_count = 0;
\r
386 char * tmp = audios;
\r
389 if( *tmp < '0' || *tmp > '9' )
\r
391 /* Skip non numeric char */
\r
395 job->audios[audio_count++] =
\r
396 strtol( tmp, &tmp, 0 ) - 1;
\r
398 job->audios[audio_count] = -1;
\r
402 job->audios[0] = -1;
\r
411 job->abitrate = abitrate;
\r
415 job->acodec = acodec;
\r
420 job->vbitrate = hb_calc_bitrate( job, size );
\r
421 fprintf( stderr, "Calculated bitrate: %d kbps\n",
\r
427 job->subtitle = sub - 1;
\r
434 job->file = strdup( output );
\r
441 if (x264opts != NULL && *x264opts != '\0' )
\r
443 hb_log("Applying the following x264 options: %s", x264opts);
\r
444 job->x264opts = x264opts;
\r
446 else /*avoids a bus error crash when options aren't specified*/
\r
448 job->x264opts = NULL;
\r
451 job->maxWidth = maxWidth;
\r
453 job->maxHeight = maxHeight;
\r
460 job->x264opts = x264opts2;
\r
472 #define p s.param.working
\r
473 case HB_STATE_WORKING:
\r
474 fprintf( stderr, "\rEncoding: task %d of %d, %.2f %%",
\r
475 p.job_cur, p.job_count, 100.0 * p.progress );
\r
476 if( p.seconds > -1 )
\r
478 fprintf( stderr, " (%.2f fps, avg %.2f fps, ETA "
\r
479 "%02dh%02dm%02ds)", p.rate_cur, p.rate_avg,
\r
480 p.hours, p.minutes, p.seconds );
\r
485 #define p s.param.muxing
\r
486 case HB_STATE_MUXING:
\r
488 fprintf( stderr, "\rMuxing: %.2f %%", 100.0 * p.progress );
\r
493 #define p s.param.workdone
\r
494 case HB_STATE_WORKDONE:
\r
495 /* Print error if any, then exit */
\r
498 case HB_ERROR_NONE:
\r
499 fprintf( stderr, "\nRip done!\n" );
\r
501 case HB_ERROR_CANCELED:
\r
502 fprintf( stderr, "\nRip canceled.\n" );
\r
505 fprintf( stderr, "\nRip failed (error %x).\n",
\r
515 /****************************************************************************
\r
517 ****************************************************************************/
\r
518 static volatile int64_t i_die_date = 0;
\r
519 void SigHandler( int i_signal )
\r
524 i_die_date = hb_get_date();
\r
525 fprintf( stderr, "Signal %d received, terminating - do it "
\r
526 "again in case it gets stuck\n", i_signal );
\r
528 else if( i_die_date + 500 < hb_get_date() )
\r
530 fprintf( stderr, "Dying badly, files might remain in your /tmp\n" );
\r
535 /****************************************************************************
\r
537 ****************************************************************************/
\r
538 static void ShowHelp()
\r
543 "Syntax: HandBrakeCLI [options] -i <device> -o <file>\n"
\r
545 "### General Handbrake Options------------------------------------------------\n\n"
\r
546 " -h, --help Print help\n"
\r
547 " -u, --update Check for updates and exit\n"
\r
548 " -v, --verbose Be verbose\n"
\r
549 " -C, --cpu Set CPU count (default: autodetected)\n"
\r
552 "### Source Options-----------------------------------------------------------\n\n"
\r
553 " -i, --input <string> Set input device\n"
\r
554 " -t, --title <number> Select a title to encode (0 to scan only,\n"
\r
556 " -c, --chapters <string> Select chapters (e.g. \"1-3\" for chapters\n"
\r
557 " 1 to 3, or \"3\" for chapter 3 only,\n"
\r
558 " default: all chapters)\n"
\r
561 "### Destination Options------------------------------------------------------\n\n"
\r
562 " -o, --output <string> Set output file name\n"
\r
563 " -f, --format <string> Set output format (avi/mp4/ogm, default:\n"
\r
564 " autodetected from file name)\n"
\r
567 "### Picture Settings---------------------------------------------------------\n\n"
\r
568 " -w, --width <number> Set picture width\n"
\r
569 " -l, --height <number> Set picture height\n"
\r
570 " --crop <T:B:L:R> Set cropping values (default: autocrop)\n"
\r
571 " -Y, --maxHeight <#> Set maximum height\n"
\r
572 " -X, --maxWidth <#> Set maximum width\n"
\r
573 " -s, --subtitle <number> Select subtitle (default: none)\n"
\r
574 " -m, --markers Add chapter markers (mp4 output format only)\n"
\r
577 "### Video Options------------------------------------------------------------\n\n"
\r
578 " -e, --encoder <string> Set video library encoder (ffmpeg,xvid,\n"
\r
579 " x264,x264b13,x264b30 default: ffmpeg)\n"
\r
580 " -q, --quality <float> Set video quality (0.0..1.0)\n"
\r
581 " -Q, --crf Use with -q for CRF instead of CQP\n"
\r
582 " -S, --size <MB> Set target size\n"
\r
583 " -b, --vb <kb/s> Set video bitrate (default: 1000)\n"
\r
584 " -r, --rate Set video framerate (" );
\r
585 for( i = 0; i < hb_video_rates_count; i++ )
\r
587 fprintf( stderr, hb_video_rates[i].string );
\r
588 if( i != hb_video_rates_count - 1 )
\r
589 fprintf( stderr, "/" );
\r
591 fprintf( stderr, ")\n"
\r
593 " -2, --two-pass Use two-pass mode\n"
\r
594 " -d, --deinterlace Deinterlace video\n"
\r
595 " -g, --grayscale Grayscale encoding\n"
\r
596 " -p, --pixelratio Store pixel aspect ratio in video stream\n"
\r
601 "### Audio Options-----------------------------------------------------------\n\n"
\r
602 " -E, --aencoder <string> Set audio encoder (faac/lame/vorbis/ac3, ac3\n"
\r
603 " meaning passthrough, default: guessed)\n"
\r
604 " -B, --ab <kb/s> Set audio bitrate (default: 128)\n"
\r
605 " -a, --audio <string> Select audio channel(s) (\"none\" for no \n"
\r
606 " audio, default: first one)\n"
\r
607 " -6, --surround Export 5.1 surround as 6-channel AAC\n"
\r
608 " -R, --arate Set audio samplerate (" );
\r
609 for( i = 0; i < hb_audio_rates_count; i++ )
\r
611 fprintf( stderr, hb_audio_rates[i].string );
\r
612 if( i != hb_audio_rates_count - 1 )
\r
613 fprintf( stderr, "/" );
\r
615 fprintf( stderr, " kHz)\n"
\r
622 "### Advanced H264 Options----------------------------------------------------\n\n"
\r
623 " -x, --x264opts <string> Specify advanced x264 options in the\n"
\r
624 " same style as mencoder:\n"
\r
625 " option1=value1:option2=value2\n" );
\r
628 /****************************************************************************
\r
630 ****************************************************************************/
\r
631 static int ParseOptions( int argc, char ** argv )
\r
635 static struct option long_options[] =
\r
637 { "help", no_argument, NULL, 'h' },
\r
638 { "update", no_argument, NULL, 'u' },
\r
639 { "verbose", no_argument, NULL, 'v' },
\r
640 { "cpu", required_argument, NULL, 'C' },
\r
642 { "format", required_argument, NULL, 'f' },
\r
643 { "input", required_argument, NULL, 'i' },
\r
644 { "output", required_argument, NULL, 'o' },
\r
646 { "title", required_argument, NULL, 't' },
\r
647 { "chapters", required_argument, NULL, 'c' },
\r
648 { "markers", no_argument, NULL, 'm' },
\r
649 { "audio", required_argument, NULL, 'a' },
\r
650 { "surround", no_argument, NULL, '6' },
\r
651 { "subtitle", required_argument, NULL, 's' },
\r
653 { "encoder", required_argument, NULL, 'e' },
\r
654 { "aencoder", required_argument, NULL, 'E' },
\r
655 { "two-pass", no_argument, NULL, '2' },
\r
656 { "deinterlace", no_argument, NULL, 'd' },
\r
657 { "grayscale", no_argument, NULL, 'g' },
\r
658 { "pixelratio", no_argument, NULL, 'p' },
\r
659 { "width", required_argument, NULL, 'w' },
\r
660 { "height", required_argument, NULL, 'l' },
\r
661 { "crop", required_argument, NULL, 'n' },
\r
663 { "vb", required_argument, NULL, 'b' },
\r
664 { "quality", required_argument, NULL, 'q' },
\r
665 { "size", required_argument, NULL, 'S' },
\r
666 { "ab", required_argument, NULL, 'B' },
\r
667 { "rate", required_argument, NULL, 'r' },
\r
668 { "arate", required_argument, NULL, 'R' },
\r
669 { "crf", no_argument, NULL, 'Q' },
\r
670 { "x264opts", required_argument, NULL, 'x' },
\r
671 { "maxHeight", required_argument, NULL, 'Y' },
\r
672 { "maxWidth", required_argument, NULL, 'X' },
\r
677 int option_index = 0;
\r
680 c = getopt_long( argc, argv,
\r
681 "hvuC:f:i:o:t:c:ma:6s:e:E:2dgpw:l:n:b:q:S:B:r:R:Qx:Y:X:",
\r
682 long_options, &option_index );
\r
697 debug = HB_DEBUG_ALL;
\r
700 cpu = atoi( optarg );
\r
704 format = strdup( optarg );
\r
707 input = strdup( optarg );
\r
710 output = strdup( optarg );
\r
714 titleindex = atoi( optarg );
\r
719 if( sscanf( optarg, "%d-%d", &start, &end ) == 2 )
\r
721 chapter_start = start;
\r
724 else if( sscanf( optarg, "%d", &start ) == 1 )
\r
726 chapter_start = start;
\r
727 chapter_end = chapter_start;
\r
731 fprintf( stderr, "chapters: invalid syntax (%s)\n",
\r
738 chapter_markers = 1;
\r
741 audios = strdup( optarg );
\r
747 sub = atoi( optarg );
\r
763 if( !strcasecmp( optarg, "ffmpeg" ) )
\r
765 vcodec = HB_VCODEC_FFMPEG;
\r
767 else if( !strcasecmp( optarg, "xvid" ) )
\r
769 vcodec = HB_VCODEC_XVID;
\r
771 else if( !strcasecmp( optarg, "x264" ) )
\r
773 vcodec = HB_VCODEC_X264;
\r
775 else if( !strcasecmp( optarg, "x264b13" ) )
\r
777 vcodec = HB_VCODEC_X264;
\r
780 else if( !strcasecmp( optarg, "x264b30" ) )
\r
782 vcodec = HB_VCODEC_X264;
\r
787 fprintf( stderr, "invalid codec (%s)\n", optarg );
\r
792 if( !strcasecmp( optarg, "ac3" ) )
\r
794 acodec = HB_ACODEC_AC3;
\r
796 else if( !strcasecmp( optarg, "lame" ) )
\r
798 acodec = HB_ACODEC_LAME;
\r
802 width = atoi( optarg );
\r
805 height = atoi( optarg );
\r
810 char * tmp = optarg;
\r
811 for( i = 0; i < 4; i++ )
\r
815 crop[i] = strtol( tmp, &tmp, 0 );
\r
824 for( i = 0; i < hb_video_rates_count; i++ )
\r
826 if( !strcmp( optarg, hb_video_rates[i].string ) )
\r
828 vrate = hb_video_rates[i].rate;
\r
834 fprintf( stderr, "invalid framerate %s\n", optarg );
\r
842 for( i = 0; i < hb_audio_rates_count; i++ )
\r
844 if( !strcmp( optarg, hb_audio_rates[i].string ) )
\r
846 arate = hb_audio_rates[i].rate;
\r
852 fprintf( stderr, "invalid framerate %s\n", optarg );
\r
857 vbitrate = atoi( optarg );
\r
860 vquality = atof( optarg );
\r
863 size = atoi( optarg );
\r
866 abitrate = atoi( optarg );
\r
872 x264opts = strdup( optarg );
\r
873 x264opts2 = strdup( optarg );
\r
876 maxHeight = atoi( optarg );
\r
879 maxWidth = atoi (optarg );
\r
883 fprintf( stderr, "unknown option (%s)\n", argv[optind] );
\r
891 static int CheckOptions( int argc, char ** argv )
\r
898 if( input == NULL || *input == '\0' )
\r
900 fprintf( stderr, "Missing input device. Run %s --help for "
\r
901 "syntax.\n", argv[0] );
\r
906 if( titleindex > 0 )
\r
908 if( output == NULL || *output == '\0' )
\r
910 fprintf( stderr, "Missing output file name. Run %s --help "
\r
911 "for syntax.\n", argv[0] );
\r
917 char * p = strrchr( output, '.' );
\r
920 if( p && !strcasecmp( p, ".avi" ) )
\r
924 else if( p && ( !strcasecmp( p, ".mp4" ) ||
\r
925 !strcasecmp( p, ".m4v" ) ) )
\r
927 if ( h264_30 == 1 )
\r
932 else if( p && ( !strcasecmp( p, ".ogm" ) ||
\r
933 !strcasecmp( p, ".ogg" ) ) )
\r
939 fprintf( stderr, "Output format couldn't be guessed "
\r
940 "from file name, using default.\n" );
\r
944 else if( !strcasecmp( format, "avi" ) )
\r
948 else if( !strcasecmp( format, "mp4" ) )
\r
955 else if( !strcasecmp( format, "ogm" ) ||
\r
956 !strcasecmp( format, "ogg" ) )
\r
962 fprintf( stderr, "Invalid output format (%s). Possible "
\r
963 "choices are avi, mp4 and ogm\n.", format );
\r
969 if( mux == HB_MUX_MP4 || mux == HB_MUX_IPOD )
\r
971 acodec = HB_ACODEC_FAAC;
\r
973 else if( mux == HB_MUX_AVI )
\r
975 acodec = HB_ACODEC_LAME;
\r
977 else if( mux == HB_MUX_OGM )
\r
979 acodec = HB_ACODEC_VORBIS;
\r
983 if (acodec != HB_ACODEC_FAAC && acodec != HB_ACODEC_VORBIS)
\r
985 /* only attempt 5.1 export if exporting to AAC or Vorbis */
\r