using System; using System.IO; using CUETools.Codecs; using CUETools.Processor; using CUETools.Processor.Settings; namespace CUETools.Converter { class Program { static void Usage() { Console.Error.WriteLine("Usage : CUETools.Converter.exe [options] "); Console.Error.WriteLine(); Console.Error.WriteLine("Options:"); Console.Error.WriteLine(); Console.Error.WriteLine(" --lossy Use lossy encoder."); Console.Error.WriteLine(" --lossless Use lossless encoder (default)."); Console.Error.WriteLine(" -p # Padding bytes."); Console.Error.WriteLine(" -m Encoder mode (0..8 for flac, V0..V9 for mp3, etc)"); Console.Error.WriteLine(); } static int Main(string[] args) { bool ok = true; string sourceFile = null, destFile = null; int padding = 8192; string encoderMode = null; AudioEncoderType audioEncoderType = AudioEncoderType.NoAudio; for (int arg = 0; arg < args.Length; arg++) { if (args[arg].Length == 0) ok = false; else if ((args[arg] == "-p" || args[arg] == "--padding") && ++arg < args.Length) ok = int.TryParse(args[arg], out padding); else if ((args[arg] == "-m" || args[arg] == "--mode") && ++arg < args.Length) encoderMode = args[arg]; else if (args[arg] == "--lossy") audioEncoderType = AudioEncoderType.Lossy; else if (args[arg] == "--lossless") audioEncoderType = AudioEncoderType.Lossless; else if ((args[arg][0] != '-' || args[arg] == "-") && sourceFile == null) sourceFile = args[arg]; else if ((args[arg][0] != '-' || args[arg] == "-") && sourceFile != null && destFile == null) destFile = args[arg]; else ok = false; if (!ok) break; } Console.Error.WriteLine("CUETools.Converter, Copyright (C) 2009-13 Grigory Chudov."); Console.Error.WriteLine("This is free software under the GNU GPLv3+ license; There is NO WARRANTY, to"); Console.Error.WriteLine("the extent permitted by law. for details."); if (!ok || sourceFile == null || destFile == null) { Usage(); return 22; } if (destFile != "-" && File.Exists(destFile)) { Console.Error.WriteLine("Error: file already exists."); return 17; } string extension = destFile == "-" ? ".wav" : Path.GetExtension(destFile).ToLower(); DateTime start = DateTime.Now; TimeSpan lastPrint = TimeSpan.FromMilliseconds(0); CUEConfig config = new CUEConfig(); SettingsReader sr = new SettingsReader("CUE Tools", "settings.txt", null); config.Load(sr); #if !DEBUG try #endif { IAudioSource audioSource = null; IAudioDest audioDest = null; TagLib.UserDefined.AdditionalFileTypes.Config = config; TagLib.File sourceInfo = sourceFile == "-" ? null : TagLib.File.Create(new TagLib.File.LocalFileAbstraction(sourceFile)); try { audioSource = sourceFile == "-" ? new WAVReader("", Console.OpenStandardInput()) : AudioReadWrite.GetAudioSource(sourceFile, null, config); AudioBuffer buff = new AudioBuffer(audioSource, 0x10000); Console.Error.WriteLine("Filename : {0}", sourceFile); Console.Error.WriteLine("File Info : {0}kHz; {1} channel; {2} bit; {3}", audioSource.PCM.SampleRate, audioSource.PCM.ChannelCount, audioSource.PCM.BitsPerSample, TimeSpan.FromSeconds(audioSource.Length * 1.0 / audioSource.PCM.SampleRate)); CUEToolsFormat fmt; if (!extension.StartsWith(".") || !config.formats.TryGetValue(extension.Substring(1), out fmt)) throw new Exception("Unsupported audio type: " + destFile); CUEToolsUDC encoder = audioEncoderType == AudioEncoderType.Lossless ? fmt.encoderLossless : audioEncoderType == AudioEncoderType.Lossy ? fmt.encoderLossy : fmt.encoderLossless != null ? fmt.encoderLossless : fmt.encoderLossy; if (encoder == null) throw new Exception("Encoder available for format " + extension + ": " + (fmt.encoderLossless != null ? fmt.encoderLossless.Name + " (lossless)" : fmt.encoderLossy != null ? fmt.encoderLossy.Name + " (lossy)" : "none")); var settings = encoder.settings.Clone(); settings.PCM = audioSource.PCM; settings.Padding = padding; settings.EncoderMode = encoderMode ?? settings.EncoderMode; settings.Validate(); object o = null; try { o = destFile == "-" ? Activator.CreateInstance(encoder.type, "", Console.OpenStandardOutput(), settings) : Activator.CreateInstance(encoder.type, destFile, settings); } catch (System.Reflection.TargetInvocationException ex) { throw ex.InnerException; } if (o == null || !(o is IAudioDest)) throw new Exception("Unsupported audio type: " + destFile + ": " + encoder.type.FullName); audioDest = o as IAudioDest; audioDest.FinalSampleCount = audioSource.Length; bool keepRunning = true; Console.CancelKeyPress += delegate(object sender, ConsoleCancelEventArgs e) { keepRunning = false; if (e.SpecialKey == ConsoleSpecialKey.ControlC) e.Cancel = true; else audioDest.Delete(); }; while (audioSource.Read(buff, -1) != 0) { audioDest.Write(buff); TimeSpan elapsed = DateTime.Now - start; if ((elapsed - lastPrint).TotalMilliseconds > 60) { long length = audioSource.Length; if (length < 0 && sourceInfo != null) length = (long)(sourceInfo.Properties.Duration.TotalMilliseconds * audioSource.PCM.SampleRate / 1000); if (length < audioSource.Position) length = audioSource.Position; if (length < 1) length = 1; Console.Error.Write("\rProgress : {0:00}%; {1:0.00}x; {2}/{3}", 100.0 * audioSource.Position / length, audioSource.Position / elapsed.TotalSeconds / audioSource.PCM.SampleRate, elapsed, TimeSpan.FromMilliseconds(elapsed.TotalMilliseconds / audioSource.Position * length) ); lastPrint = elapsed; } if (!keepRunning) throw new Exception("Aborted"); } TimeSpan totalElapsed = DateTime.Now - start; Console.Error.Write("\r \r"); Console.Error.WriteLine("Results : {0:0.00}x; {1}", audioSource.Position / totalElapsed.TotalSeconds / audioSource.PCM.SampleRate, totalElapsed ); } catch (Exception ex) { if (audioSource != null) audioSource.Close(); if (audioDest != null) audioDest.Delete(); throw ex; } audioSource.Close(); audioDest.Close(); if (sourceFile != "-" && destFile != "-") { TagLib.File destInfo = destFile == "-" ? null : TagLib.File.Create(new TagLib.File.LocalFileAbstraction(destFile)); if (Tagging.UpdateTags(destInfo, Tagging.Analyze(sourceInfo), config)) { sourceInfo.Tag.CopyTo(destInfo.Tag, true); destInfo.Tag.Pictures = sourceInfo.Tag.Pictures; destInfo.Save(); } } } #if !DEBUG catch (Exception ex) { Console.Error.Write("\r \r"); Console.Error.WriteLine("Error : {0}", ex.Message); return 1; //Console.WriteLine("{0}", ex.StackTrace); } #endif return 0; } } }